1

我的任务是实现一个 JTable,其中一个列具有以下外观: 在此处输入图像描述

它应该包含 4 个元素:
1. 一个 JCombobox,它控制其他两个的启用状态:它们被启用为“手动”状态,并锁定其他状态的某些数值(让我们称其他组合状态:'first ','第二','第三')。
2. 一个 JTextField,当启用时,它控制滑块的值。
3. 一个读取“%”但什么都不做的 JLabel。
4. s JSlider,当启用时,它控制 JTextField 值(这种相互控制甚至存在吗?)。

我爬遍了整个网络,寻找实现这一点的方法。大多数例子都太肤浅了,包括这个类似的问题

在我问这个问题之前,我曾经历过 Cell Rendering/fireEditingStopped 等方面的折磨,但遗憾的是没有简单的短代码可以显示。

谁能告诉我具体的指导方针或代码片段可以让我走上正确的道路吗?提前致谢。

4

4 回答 4

6

这是所有单元格元素的单个渲染器/编辑器还是每个元素的多个 Ren'/Ed'?

我不确定这个问题,但这是一个从TableCellRendererand检索数据的简单示例TableCellEditor

在此处输入图像描述

import java.awt.*;
import java.awt.event.*;
import java.util.*;
import javax.swing.*;
import javax.swing.event.*;
import javax.swing.table.*;

public class MultipleComponentCellTest {
  private final String[] columnNames = {"Band Type"};
  private final Object[][] data = {
    {new BandType("manual",  50)},
    {new BandType("locked", 100)},
    {new BandType("manual",  32)},
    {new BandType("locked",   0)},
  };
  private final TableModel model = new DefaultTableModel(data, columnNames) {
    @Override public Class<?> getColumnClass(int column) {
      return BandType.class;
    }
  };
  private final JTable table = new JTable(model);
  public JComponent makeUI() {
    table.setRowHeight(42);
    table.setDefaultRenderer(BandType.class, new BandTypeRenderer());
    table.setDefaultEditor(BandType.class, new BandTypeEditor());
    return new JScrollPane(table);
  }
  public static void main(String... args) {
    EventQueue.invokeLater(new Runnable() {
      @Override public void run() {
        createAndShowGUI();
      }
    });
  }
  public static void createAndShowGUI() {
    try {
      for (UIManager.LookAndFeelInfo laf: UIManager.getInstalledLookAndFeels()) {
        if ("Nimbus".equals(laf.getName())) {
          UIManager.setLookAndFeel(laf.getClassName());
        }
      }
    } catch (ClassNotFoundException | InstantiationException
           | IllegalAccessException | UnsupportedLookAndFeelException ex) {
      ex.printStackTrace();
    }
    JFrame f = new JFrame();
    f.setDefaultCloseOperation(WindowConstants.EXIT_ON_CLOSE);
    f.getContentPane().add(new MultipleComponentCellTest().makeUI());
    f.setSize(320, 240);
    f.setLocationRelativeTo(null);
    f.setVisible(true);
  }
}

class BandType {
  public final String state;
  public final int value;
  public BandType(String state, int value) {
    this.state = state;
    this.value = value;
  }
}

class BandTypePanel extends JPanel {
  private static String DEFAULT = "0";
  public final JSlider slider = new JSlider(0, 100);
  public final JTextField textField = new JTextField(3);
  public final JComboBox<String> comboBox = new JComboBox<>(
      new String[] {"manual", "locked"});
  public BandTypePanel() {
    super(new BorderLayout(5, 5));
    setBorder(BorderFactory.createEmptyBorder(5, 5, 5, 5));

    comboBox.addItemListener(new ItemListener() {
      @Override public void itemStateChanged(ItemEvent e) {
        if (e.getStateChange() == ItemEvent.SELECTED) {
          boolean f = "manual".equals(e.getItem());
          slider.setEnabled(f);
          textField.setEnabled(f);
        }
      }
    });

    textField.setEditable(false);
    textField.setHorizontalAlignment(JTextField.RIGHT);

    slider.setOpaque(false);
    slider.setFocusable(false);
    slider.getModel().addChangeListener(new ChangeListener() {
      @Override public void stateChanged(ChangeEvent e) {
        BoundedRangeModel m = (BoundedRangeModel) e.getSource();
        textField.setText(Objects.toString(m.getValue(), DEFAULT));
      }
    });

    JPanel p = new JPanel();
    p.setOpaque(false);
    p.add(comboBox);
    p.add(textField);
    p.add(new JLabel("%"));

    add(p, BorderLayout.WEST);
    add(slider);
  }
  public void updateValue(BandType bt) {
    comboBox.setSelectedItem(bt.state);
    slider.setValue(bt.value);
    textField.setText(Objects.toString(bt.value, DEFAULT));
  }
}

class BandTypeRenderer extends BandTypePanel implements TableCellRenderer {
  public BandTypeRenderer() {
    super();
    setName("Table.cellRenderer");
  }
  @Override public Component getTableCellRendererComponent(
      JTable table, Object value, boolean isSelected, boolean hasFocus,
      int row, int column) {
    setBackground(isSelected ? table.getSelectionBackground() : table.getBackground());
    if (value instanceof BandType) {
      updateValue((BandType) value);
    }
    return this;
  }
}

class BandTypeEditor extends BandTypePanel implements TableCellEditor {
  protected transient ChangeEvent changeEvent;
  @Override public Component getTableCellEditorComponent(
    JTable table, Object value, boolean isSelected, int row, int column) {
    this.setBackground(table.getSelectionBackground());
    if (value instanceof BandType) {
      updateValue((BandType) value);
    }
    return this;
  }
  @Override public Object getCellEditorValue() {
    return new BandType(comboBox.getSelectedItem().toString(), slider.getValue());
  }
  //Copied from AbstractCellEditor
  //protected EventListenerList listenerList = new EventListenerList();
  @Override public boolean isCellEditable(EventObject e) {
    return true;
  }
  @Override public boolean shouldSelectCell(EventObject anEvent) {
    return true;
  }
  @Override public boolean stopCellEditing() {
    fireEditingStopped();
    return true;
  }
  @Override public void cancelCellEditing() {
    fireEditingCanceled();
  }
  @Override public void addCellEditorListener(CellEditorListener l) {
    listenerList.add(CellEditorListener.class, l);
  }
  @Override public void removeCellEditorListener(CellEditorListener l) {
    listenerList.remove(CellEditorListener.class, l);
  }
  public CellEditorListener[] getCellEditorListeners() {
    return listenerList.getListeners(CellEditorListener.class);
  }
  protected void fireEditingStopped() {
    // Guaranteed to return a non-null array
    Object[] listeners = listenerList.getListenerList();
    // Process the listeners last to first, notifying
    // those that are interested in this event
    for (int i = listeners.length - 2; i >= 0; i -= 2) {
      if (listeners[i] == CellEditorListener.class) {
        // Lazily create the event:
        if (Objects.isNull(changeEvent)) {
          changeEvent = new ChangeEvent(this);
        }
        ((CellEditorListener) listeners[i + 1]).editingStopped(changeEvent);
      }
    }
  }
  protected void fireEditingCanceled() {
    // Guaranteed to return a non-null array
    Object[] listeners = listenerList.getListenerList();
    // Process the listeners last to first, notifying
    // those that are interested in this event
    for (int i = listeners.length - 2; i >= 0; i -= 2) {
      if (listeners[i] == CellEditorListener.class) {
        // Lazily create the event:
        if (Objects.isNull(changeEvent)) {
          changeEvent = new ChangeEvent(this);
        }
        ((CellEditorListener) listeners[i + 1]).editingCanceled(changeEvent);
      }
    }
  }
}
于 2015-06-17T09:02:28.770 回答
3

对于任何可编辑的组件,您都需要aTableCellRenderer a此处TableCellEditor显示了一个一般示例。

  • 默认渲染器将适用于“%”,如此String所示;或者,您可能希望在相邻列的渲染器中显示它。

  • 默认渲染器和编辑器适用于表示百分比的Double或模型值,如此Float所示。

  • ADefaultCellEditor将适用于组合框,如下所示;你可以装饰渲染器,如图所示

  • 尝试SliderColumnSpinnerColumn,显示在此处,用于最后一列;检查此处看到的示例以获取更多信息。

于 2015-06-16T15:07:15.457 回答
0

您需要实现 TableCellRenderer。因此,您可以将多个组件添加到同一个单元格。在这里您可以找到更多信息http://www.coderanch.com/t/614645/GUI/java/adding-components-cell-jtable

于 2015-06-16T13:09:30.583 回答
0

如果您所做的一切都在寻找,那么这里的示例就可以工作。当您尝试与之交互时,它会崩溃。我是盲人,除了使用鼠标外,还有很多方法可以访问计算机。当我使用这个表时,我看到的只是 4 行,带类型作为单元格 1 的名称。我无法进入组合框或任何控件。尝试仅使用键盘使用此波段类型表,看看您是否可以使用组合框执行任何操作,或者是否可以选择编辑框。如果您想看看这到底有多糟糕,请下载 NVDA 免费屏幕阅读器,然后只使用键盘和屏幕阅读器尝试表格。可以在这里找到:https ://www.nvaccess.org/download/

我目前正在使用 Arduino IDE 来尝试修复这样的表格,但到目前为止,我还无法让所有组件都集中注意力。重要的是人们要确保当他们创建这样的东西时,如果您使用键盘导航以及鼠标甚至触摸,焦点会起作用。我对任何能让这张桌子为 NVDA 工作的人都非常感兴趣。如果我先到那里,我将发布 Panel Cell 多组件单元的正确实现,但这个不是。

于 2018-12-21T03:03:25.333 回答