- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我的最终目标是拥有两个 JList
,用户可以在它们之间来回移动元素。我使用 TreeSet
以便按字母顺序插入元素。这是一个直观的表示:
下面是我迄今为止的代码,位于 SSCCE 中。它大部分工作正常,但有时我会收到 ArrayOutOfBoundsException 异常。问题与程序认为选择的元素比实际选择的元素多有关。重现问题的一种方法是选择左侧的两个元素,然后按两次右侧按钮。
import java.awt.Dimension;
import java.awt.GridBagLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.*;
import javax.swing.*;
public class GUI extends JFrame {
private GridBagLayout gridBag = new GridBagLayout();
private static final String[] ALL_STRINGS = { "B", "A", "C" };
private JButton leftButton = new JButton("<");
private JButton rightButton = new JButton(">");
private StringListModel listModel = new StringListModel(Arrays.asList(ALL_STRINGS));
private StringListModel queueModel = new StringListModel();
private JList<String> list = new JList<String>(listModel);
private JList<String> queue = new JList<String>(queueModel);
public GUI() {
setWindowProperties();
addComponents();
}
private void setWindowProperties() {
setLayout(gridBag);
setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
setSize(new Dimension(300, 200));
setTitle("JList, ListModel, and TreeSet");
setResizable(false);
setLocationRelativeTo(null);
}
private void addComponents() {
leftButton.addActionListener(new QueueListener());
rightButton.addActionListener(new QueueListener());
JScrollPane listScroll = new JScrollPane(list);
JScrollPane queueScroll = new JScrollPane(queue);
Dimension scrollSize = new Dimension(50, 100);
listScroll.setPreferredSize(scrollSize);
queueScroll.setPreferredSize(scrollSize);
add(listScroll);
add(leftButton);
add(rightButton);
add(queueScroll);
}
private class QueueListener implements ActionListener {
private JButton button;
@Override
public void actionPerformed(ActionEvent e) {
button = (JButton) e.getSource();
if (button.equals(leftButton)) {
removeFromQueue();
} else if (button.equals(rightButton)) {
addToQueue();
}
}
private void removeFromQueue() {
List<String> Strings = queue.getSelectedValuesList();
queueModel.removeAll(Strings);
listModel.addAll(Strings);
}
private void addToQueue() {
List<String> Strings = list.getSelectedValuesList();
listModel.removeAll(Strings);
queueModel.addAll(Strings);
}
}
private class StringListModel extends DefaultListModel<String> {
private TreeSet<String> model = new TreeSet<String>();
public StringListModel() {
}
public StringListModel(List<String> Strings) {
addAll(Strings);
}
public int getSize() {
return model.size();
}
public String getElementAt(int index) {
return (String) model.toArray()[index];
}
public void add(String String) {
if (model.add(String)) {
fireContentsChanged(this, 0, getSize());
}
}
public void addAll(List<String> quets) {
for (String String : quets) {
model.add(String);
}
fireContentsChanged(this, 0, getSize());
}
public void clear() {
model.clear();
fireContentsChanged(this, 0, getSize());
}
public boolean contains(Object element) {
return model.contains(element);
}
public String firstElement() {
return model.first();
}
public Iterator iterator() {
return model.iterator();
}
public String lastElement() {
return model.last();
}
public void removeAll(Collection<?> elements) {
for (Object element : elements) {
removeElement(element);
}
fireContentsChanged(this, 0, getSize());
}
public boolean removeElement(Object element) {
boolean removed = model.remove(element);
if (removed) {
fireContentsChanged(this, 0, getSize());
}
return removed;
}
}
public static void main(String[] args) {
new GUI().setVisible(true);
}
}
最佳答案
这个实现似乎有效。请注意,我将 TreeSet
更改为简单的 List
。缺点是插入/删除会慢一些,但至少我可以触发正确的事件。鉴于它是一个 UI 小部件,我认为这是可以接受的,因为底层模型永远不会很大(否则 UI 将变得不可用)。
我从 AbstractListModel
扩展,但您也可以从 DefaultListModel
扩展。在这种情况下,您可能希望确保 addElement
方法计算正确的索引并调用 super.add(calculatedIndex, element)
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.GridBagLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.util.*;
import javax.swing.*;
public class GUI extends JFrame {
private GridBagLayout gridBag = new GridBagLayout();
private static final String[] ALL_STRINGS = {"B", "A", "C"};
private JButton leftButton = new JButton( "<" );
private JButton rightButton = new JButton( ">" );
private StringListModel listModel = new StringListModel( Arrays.asList( ALL_STRINGS ) );
private StringListModel queueModel = new StringListModel();
private JList<String> list = new JList<String>( listModel );
private JList<String> queue = new JList<String>( queueModel );
public GUI() {
setWindowProperties();
addComponents();
}
private void setWindowProperties() {
setLayout( gridBag );
setDefaultCloseOperation( JFrame.EXIT_ON_CLOSE );
setSize( new Dimension( 300, 200 ) );
setTitle( "JList, ListModel, and TreeSet" );
setResizable( false );
setLocationRelativeTo( null );
}
private void addComponents() {
leftButton.addActionListener( new QueueListener() );
rightButton.addActionListener( new QueueListener() );
JScrollPane listScroll = new JScrollPane( list );
JScrollPane queueScroll = new JScrollPane( queue );
Dimension scrollSize = new Dimension( 50, 100 );
listScroll.setPreferredSize( scrollSize );
queueScroll.setPreferredSize( scrollSize );
add( listScroll );
add( leftButton );
add( rightButton );
add( queueScroll );
}
private class QueueListener implements ActionListener {
private JButton button;
@Override
public void actionPerformed( ActionEvent e ) {
button = ( JButton ) e.getSource();
if ( button.equals( leftButton ) ) {
removeFromQueue();
}
else if ( button.equals( rightButton ) ) {
addToQueue();
}
}
private void removeFromQueue() {
List<String> Strings = queue.getSelectedValuesList();
queueModel.removeAll( Strings );
listModel.addAll( Strings );
}
private void addToQueue() {
List<String> Strings = list.getSelectedValuesList();
listModel.removeAll( Strings );
queueModel.addAll( Strings );
}
}
private class StringListModel extends AbstractListModel<String> {
private List<String> model = new ArrayList<>();
public StringListModel() {
}
public StringListModel( List<String> strings ) {
addAll( strings );
}
@Override
public int getSize() {
return model.size();
}
@Override
public String getElementAt( int index ) {
return model.toArray( new String[model.size()])[index];
}
public void addAll( Collection<String> strings ){
for ( String string : strings ) {
add( string );
}
}
public void add( String string ){
int index = calculateIndex( string );
model.add( index, string );
fireIntervalAdded( this, index, index );
}
public void remove( String string ){
int index = model.indexOf( string );
if ( index != -1 ){
model.remove( index );
fireIntervalRemoved( this, index, index );
}
}
public void removeAll( Collection<String> strings ){
for ( String next : strings ) {
remove( next );
}
}
private int calculateIndex( String input ){
if ( model.size() == 0 ){
return 0;
}
int index = 0;
while ( model.get( index ).compareTo( input ) <=0 ){
index++;
if ( index == model.size() ){
return index;
}
}
return index;
}
}
public static void main( String[] args ) {
EventQueue.invokeLater( new Runnable() {
@Override
public void run() {
new GUI().setVisible( true );
}
} );
}
}
关于java - 使用 ListModel 来隐藏 TreeSet,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18560930/
我有两个 ListModel:listModelRight 和 listModelLeft。我只是想要 listModelRight = ListModelLeft,但即使这样也不起作用。我该怎么做?
我需要创建一个 ListModel , 静态地包含一个对象(字符串和 bool 值)。 如果我添加到一个空 ListModel元素通过使用 append - 一切正常。 property ListMo
所以我正在制作一个包含这段代码的程序 final ListModel listModel = new DefaultListModel(); JButton btnBuscar = new JButt
如何将菜单添加到具有适当操作的列表模型中? 我见过这样的代码: list menu: [ :menu | menu add: 'Name' action: [ "some acti
我有以下 C++ 模型结构: Manager // QAbstractListModel ↪ Slot // QAbstractListModel
当我单击一个按钮(例如 mainBtn),然后在 MealList 中选择一个字符串,然后单击另一个按钮(例如 starterBtn)时,控制台中会抛出错误。如果有人可以将我推向正确的方向并告诉我为什
我有一个 JList,它使用列表模型将文本字符串添加到 JList。我无法弄清楚如何更新选定的 listModel。我可以选择 listModel 字符串(我已验证我选择的内容是否正确返回),但我无法
我不知道为什么会发生这种情况,如果需要的话我会查看代码的其他部分,但我真的更喜欢它是其他东西...... 假设我的 JList 中有以下项目 Beer Mercedes Coca Cola 并且 st
我可以保存使用 Component.onComponent 静态创建的列表项的设置方法。但是静态创建的列表项的设置会在重新打开应用程序后生效。我想保存动态创建的列表模型的设置。我无法保存动态创建的列表
我有一个TableView动态填充 ListModel我需要在“QML 端”进行排序,最好不要替换列表中的任何元素,因为几个表信号(包括一些自定义信号)附加了相当多的逻辑。 我遇到的问题是,当表增长超
在 Codenameone 中,我使用列表作为容器渲染器来形成,在使用 hastable 解析 JSON 数据后,我在列表的项目上显示结果。该列表包含 hastable 已通过 ListModel它工
我想要一个 Swipeview,在一个 swipeview 中有一个小部件(按钮)列表。一切都必须动态创建,因为可以有一个或五个屏幕,或者一个或更多小部件。它可以通过抽屉添加(将来)。在抽屉上添加屏幕
我正在尝试创建我的第一个列表,但遇到错误。 这是我用于列表模型的模板: private ListModel getListModel() { String[] arrayOfStrin
假设我们有一个名为 JList1 的列表。 我们要为它创建一个模型,所以我们当然要使用一个ListModel对象。这就是应该做的。 DefaultListModel numbers = new Def
我想知道如何在 ListModel 中传递数组? 好的,在 QML 中,我有一个 ListView,我将其设置为 ListModel,如下所示: model: ListModel { id:
我想构建多个 ListModel,它们都将以公共(public)元素开头。我想填充另一个 ListModel 中的剩余元素。 例子:列表模型(一): COMMON A COMMON B COMMON
我有我的自定义列表模型,我在其中放置了应该显示在 QML View 上的数据。但由于某种原因,QML 中的 View 有时会正常更新,有时会使用以前的数据,有时不会执行更新。 这是我填充模型的函数 -
我在获取 ListModel 的所有元素时遇到问题,当我尝试将字符串写入文件时,它们输出如下所示的内容: [Ljava.lang.String;@79b43f[Ljava.lang.String;@7
我正在构建一个应用程序,它基本上只是一个用于显示和编辑单个对象的 GUI。由于该对象的类是使用 JAXB 创建的,它基本上是一棵有许多字符串和整数作为子项的树,但也有一些 List s。对于每个简单的
我的最终目标是拥有两个 JList,用户可以在它们之间来回移动元素。我使用 TreeSet 以便按字母顺序插入元素。这是一个直观的表示: 下面是我迄今为止的代码,位于 SSCCE 中。它大部分工作正常
我是一名优秀的程序员,十分优秀!