我在表格单元格中使用了多个TableCellRenderer
的自定义JFormattedTextField
。我使用与TableCellEditor
相同的组件。现在我需要知道用户点击的JFormattedTextField
,以及此字段中的位置(可以使用viewToModel
完成)。
使用自定义TableCellEditor
时,通过鼠标点击获取Point
的唯一方法是CellEditor
中的isCellEditable(EventObject e)
方法。 Point
给出的是父母坐标系。
anEvent位于调用组件坐标系中。
但是如何在点击的坐标上获取组件?我尝试使用findComponentAt(Point p)
,但它会为我返回null
。
以下是我测试过的一些代码:
@Override
public boolean isCellEditable(EventObject e) {
if(e instanceof MouseEvent) {
MouseEvent ev = (MouseEvent)e;
Point p = ev.getPoint();
// gives strange values
Point p3 = editor.getLocation();
// x: 0 y: 0
Point tp = ((JTable)e.getSource()).getLocation();
// these returns null
Component c1 = renderer.findComponentAt(p);
Component c2 = editor.findComponentAt(p);
System.out.println("Click at " + p + " editor at: " + p3);
}
return true;
}
组件位置editor.getLocation();
的值为y坐标提供了几乎随机的值(例如,在表格中使用5行时)。
使用TableCellEditor
和TableCellRenderer
时,如何获取用户点击的组件?
以下是一个完整的例子:
public class FormattedTableEditDemo extends JFrame {
public FormattedTableEditDemo() {
MyTableModel model = new MyTableModel();
MyTableCellEditorAndRenderer cellEditorAndRenderer =
new MyTableCellEditorAndRenderer();
JTable table = new JTable(model);
table.setDefaultRenderer(BigDecimal.class, cellEditorAndRenderer);
table.setDefaultEditor(BigDecimal.class, cellEditorAndRenderer);
table.setRowHeight(40);
add(new JScrollPane(table));
pack();
setDefaultCloseOperation(EXIT_ON_CLOSE);
setVisible(true);
}
class MyTableCellEditorAndRenderer extends AbstractCellEditor
implements TableCellEditor, TableCellRenderer {
MyCellPanel editor = new MyCellPanel();
MyCellPanel renderer = new MyCellPanel();
@Override
public Object getCellEditorValue() {
return editor.getValue();
}
@Override
public Component getTableCellRendererComponent(JTable table,
Object value, boolean isSelected, boolean hasFocus,
int row, int column) {
renderer.setValue(value);
return renderer;
}
@Override
public Component getTableCellEditorComponent(JTable table,
Object value, boolean isSelected, int row, int column) {
editor.setValue(value);
return editor;
}
@Override
public boolean shouldSelectCell(EventObject e) {
return false;
}
@Override
public boolean isCellEditable(EventObject e) {
if(e instanceof MouseEvent) {
MouseEvent ev = (MouseEvent)e;
Point p = ev.getPoint();
// gives strange values
Point p3 = editor.getLocation();
// x: 0 y: 0
Point tp = ((JTable)e.getSource()).getLocation();
// these returns null
Component c1 = renderer.findComponentAt(p);
Component c2 = editor.findComponentAt(p);
System.out.println("Click at " + p + " editor at: " + p3);
}
return true;
}
}
class MyCellPanel extends JPanel {
JFormattedTextField field1 = new JFormattedTextField();
JFormattedTextField field2 = new JFormattedTextField();
public MyCellPanel() {
field1.setColumns(8);
field2.setColumns(8);
field2.setValue(new BigDecimal("0.00"));
setLayout(new BorderLayout());
add(field1, BorderLayout.WEST);
add(Box.createHorizontalStrut(30));
add(field2, BorderLayout.EAST);
}
public Object getValue() {
return field1.getValue();
}
public void setValue(Object value) {
field1.setValue(value);
}
}
class MyTableModel extends AbstractTableModel {
List<BigDecimal> values = new ArrayList<BigDecimal>();
public MyTableModel() {
// test values
values.add(new BigDecimal("37.00"));
values.add(new BigDecimal("4305.90"));
values.add(new BigDecimal("386.04"));
values.add(new BigDecimal("3486.58"));
values.add(new BigDecimal("6546.45"));
}
@Override
public int getColumnCount() {
return 1;
}
@Override
public int getRowCount() {
return values.size();
}
@Override
public Object getValueAt(int row, int column) {
return values.get(row);
}
@Override
public boolean isCellEditable(int row, int column) {
return true;
}
@Override
public Class<?> getColumnClass(int column) {
return BigDecimal.class;
}
}
public static void main(String[] args) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
new FormattedTableEditDemo();
}
});
}
}
答案 0 :(得分:10)
我不完全确定我知道出了什么问题(只是让我知道如果我关闭了,所以我可以删除它: - )
假设您希望获得触发编辑开始的鼠标(单击/按下)下的“真实”组件,诀窍是在之后进行转换(从父级到编辑器坐标)编辑器已添加到其父级。这对于shouldSelectCell是有保证的,但对于isCellEditable(后者在之前被称为)则无法保证
树的上下文中的recent answer(应该足够相似)有一些可运行的示例。这是相关的片段:
/**
* At this point in time the editing component is added to the table (not documented!) but
* table's internal cleanup might not yet be ready
*/
@Override
public boolean shouldSelectCell(EventObject anEvent) {
if (anEvent instanceof MouseEvent) {
redirect((MouseEvent) anEvent);
}
return false;
}
private void redirect(final MouseEvent anEvent) {
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run() {
MouseEvent ev = SwingUtilities.convertMouseEvent(anEvent.getComponent(), anEvent, editor);
// at this point you have the mouse coordinates in the editor's system
// do stuff, like f.i. findComponent
....
}
});
}
答案 1 :(得分:1)
这不是您的问题的答案,而是对您看到的结果的解释:findComponentAt()
返回null
,因为“请求的点上没有子组件。” MyCellPanel
位于JTable
{{1}}上,以加快呈现速度。有一个如何使用CellRendererPane
的例子。
答案 2 :(得分:1)
您可以从点击的点获取表格的行和列。然后调用相同的渲染器的getTableCellRendererComponent方法来获取渲染器组件。然后正确的点从y减去前一行的高度,从x减去先前单元格的宽度。然后得到渲染组件的正确子项。