JList单元格中的JButton无法点击。

9
在我的Swing项目中,我有一个显示所有活动套接字的JList,并且每个单元格都有一个关闭该套接字的JButton。但是,单元格中的JButton无法点击:监听器不会触发。
我已将代码修改为最小值,如下所示。
private class ConnectionListRenderer extends JButton implements ListCellRenderer {

    public Component getListCellRendererComponent(JList jlist, Object o, int i, boolean bln, boolean bln1) {

        addActionListener(new ActionListener() {
            public void actionPerformed(ActionEvent evt) {
                //do something (close the socket in my project)
            }
        });
        return this;
    }
}


jList.setCellRenderer(new ConnectionListRenderer());

这个列表看起来不错,但是里面的按钮不能点击。我错了吗,或者说 JList 不支持在其中放置 JButton

3个回答

7

这里有一个例子,看起来似乎可以运行,但你得不到一个正常按钮点击的视觉效果。也许比我更擅长绘画的人可以改进它,模拟出视觉上按下按钮的效果。

import java.awt.*;
import java.awt.event.*;

import javax.swing.*;

/**
 * A JList of JButtons.
 */
public class JButtonListDemo implements Runnable
{
  private JList jlist;

  public static void main(String args[])
  {
    SwingUtilities.invokeLater(new JButtonListDemo());
  }

  public void run()
  {
    Object[] items = new ButtonItem[] {
        new ButtonItem("Apple"),
        new ButtonItem("Banana"),
        new ButtonItem("Carrot"),
        new ButtonItem("Date"),
        new ButtonItem("Eggplant"),
        new ButtonItem("Fig"),
        new ButtonItem("Guava"),
    };

    jlist = new JList(items);
    jlist.setCellRenderer(new ButtonListRenderer());
    jlist.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
    jlist.setVisibleRowCount(5);
    jlist.addMouseListener(new MouseAdapter()
    {
      @Override
      public void mouseClicked(MouseEvent event)
      {
        clickButtonAt(event.getPoint());
      }
    });

    JFrame frame = new JFrame();
    frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
    frame.getContentPane().add(new JScrollPane(jlist));
    frame.pack();
    frame.setLocationRelativeTo(null);
    frame.setVisible(true);
  }

  private void clickButtonAt(Point point)
  {
    int index = jlist.locationToIndex(point);
    ButtonItem item = (ButtonItem) jlist.getModel().getElementAt(index);
    item.getButton().doClick();
//    jlist.repaint(jlist.getCellBounds(index, index));
  }

  public class ButtonItem
  {
    private JButton button;

    public ButtonItem(String name)
    {
      this.button = new JButton(name);
      button.addActionListener(new ActionListener()
      {
        @Override
        public void actionPerformed(ActionEvent e)
        {
          System.out.println(button.getText() + " was clicked.");
        }
      });
    }

    public JButton getButton()
    {
      return button;
    }

    @Override
    public String toString()
    {
      return button.getText();
    }
  }

  class ButtonListRenderer extends JButton implements ListCellRenderer
  {
    public Component getListCellRendererComponent(JList comp, Object value, int index,
                                                  boolean isSelected, boolean hasFocus)
    {
      setEnabled(comp.isEnabled());
      setFont(comp.getFont());
      setText(value.toString());

      if (isSelected)
      {
        setBackground(comp.getSelectionBackground());
        setForeground(comp.getSelectionForeground());
      }
      else
      {
        setBackground(comp.getBackground());
        setForeground(comp.getForeground());
      }

      return this;
    }
  }
}

或者,您可以在JPanel上垂直布置您的JButtons(使用一个新的GridLayout(0,1)),然后将您的JPanel放在JScrollPane中,从而模拟JList的JButtons。


ButtonItem item = (ButtonItem) jlist.getModel().getElementAt(index); 会报错:无法将 String 转换为 ButtonItem :( - Dohab
@Dohab - 那么你肯定对示例进行了某些修改。我刚刚按原样尝试了上面的代码,它可以正常工作。 - splungebob

4
渲染器不是“真正的”组件,它们只是涂在父组件表面的“橡皮图章”,没有“实体”存在。
一个JList将只有一个渲染器实例,并用于将列表模型中的每个项目“盖印”到视图上。
开箱即用的JList不可编辑。

0
另一个解决方案是使用两个相邻的列表。第一个列表呈现实际的列表内容,而第二个列表呈现按钮,您可以将这两个列表添加到一个JPanel中,并使用BorderLayout (Borderlayout.CENTERBorderLayout.EAST)进行布局。将此JPanel添加到JScrollPane的视口中。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接