8

現在の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

4

3 に答える 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;
    }
  }
}

または、JButton を JPanel 上に垂直にレイアウトし (おそらく new GridLayout(0,1) を使用)、JPanel を JScrollPane に配置して、JButton の JList をモックすることもできます。

于 2012-11-19T16:51:48.360 に答える
4

レンダーは「実際の」コンポーネントではなく、親コンポーネントの表面にペイントされた「ラバースタンプ」です。それらには「物理的な」存在はありません。

AJListにはレンダリングのインスタンスが1つだけあり、これはリストモデルの各アイテムをビューに「スタンプ」するために使用されます。

箱から出して、JList編集することはできません。

于 2012-11-19T05:13:23.617 に答える