Изменить поведение Ctrl + клик на JTable
Вопрос
Есть ли простой способ манипулировать элементами управления в JTable, чтобы обеспечить различные функциональные возможности, когда нажата кнопка клавиатуры (например, кнопка CTRL) и выбрана строка? Меня попросили создать таблицу, в которой сочетание клавиш CTRL + щелчок (щелчок мыши) по строке будет отменять только выделенную строку, но никогда не выделять строку. Если пользователь CTRL + щелкает невыбранную строку, ничего не произойдет. Р>
Мне удалось создать таблицу и отключить такие функции, как CTRL + A (выбрать все), и я смог проверить, нажата ли кнопка управления при создании MouseEvent, но я не могу кажется, выяснить, как можно настроить CTRL + Click. Вот некоторый код:
package nicky;
import javax.swing.*;
import java.awt.Dimension;
import java.awt.GridLayout;
import java.awt.event.*;
public class TableTester extends JPanel {
public TableTester() {
super(new GridLayout(1,0));
final String[] columnNames = {"First Name",
"Last Name",
"Sport",
"# of Years",
"Vegetarian"};
final Object[][] data = {
{"Tom", "Roberts","Athletic", new Integer(5), new Boolean(false)},
{"Sarah", "Watt", "Football", new Integer(3), new Boolean(true)},
{"Laura", "Brown", "Swimming", new Integer(2), new Boolean(false)},
{"Simon", "Smith", "Tennis", new Integer(20), new Boolean(true)},
{"Paul", "Jones", "Rugby", new Integer(10), new Boolean(false)}
};
JTable table = new JTable(data, columnNames);
table.setPreferredScrollableViewportSize(new Dimension(500, 100));
table.setSelectionMode(ListSelectionModel.MULTIPLE_INTERVAL_SELECTION);
table.addMouseListener(new MouseListener(){
public void mouseEntered(MouseEvent me){}
public void mouseExited(MouseEvent me){}
public void mouseReleased(MouseEvent me){}
public void mouseClicked(MouseEvent me){}
public void mousePressed(MouseEvent me){
if (me.isControlDown()){
System.out.println("This is working ");
}
}
});
InputMap inputMap = table.getInputMap(JTable.WHEN_ANCESTOR_OF_FOCUSED_COMPONENT);
KeyStroke keyStroke = KeyStroke.getKeyStroke(KeyEvent.VK_A, InputEvent.CTRL_MASK);
inputMap.put(keyStroke, "none");
JScrollPane scrollPane = new JScrollPane(table);
add(scrollPane);
}
private static void createAndShowGUI() {
JFrame.setDefaultLookAndFeelDecorated(true);
JFrame frame = new JFrame("TableTester");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
TableTester newContentPane = new TableTester();
newContentPane.setOpaque(true);
frame.setContentPane(newContentPane);
frame.pack();
frame.setVisible(true);
}
public static void main(String[] args) {
javax.swing.SwingUtilities.invokeLater(new Runnable() {
public void run() {
createAndShowGUI();
}
});
}
}
В методе mousePressed я поиграл с получением всех выбранных строк из таблицы, а затем собирался проверить, была ли недавно нажатая строка в selectedRows ... Однако я не уверен, есть ли способ узнать, какая строка связана с MouseEvent. Р>
(Кроме того, я знаю, что ожидаемое поведение, такое как это, не должно быть слишком сложным, но оно должно копировать унаследованную систему в компании)
Любые идеи / предложения будут оценены! Р>
Решение
Хорошо, второй дубль (я оставил первый, так как он может заинтересовать кого-то для другого использования, кто знает? Скажите, что он есть в образовательных целях ...: -)).
Я взглянул на исходный код JTable и обнаружил, что события мыши обрабатываются внешним видом. Зная, как он обрабатывает управляющий ключ, я смог смело переопределить метод changeSelection, чтобы сделать то, что вам нужно.
Я нахожу требования немного странными (вы все еще можете использовать Shift + щелчок, нет?), Но я не знаю контекста.
class SpecialTable extends JTable
{
SpecialTable(Object[][] data, String[] columnNames)
{
super(data, columnNames);
// That's already the default
// setSelectionMode(ListSelectionModel.MULTIPLE_INTERVAL_SELECTION);
}
/**
* Called by javax.swing.plaf.basic.BasicTableUI.Handler.adjustSelection(MouseEvent)
* like: table.changeSelection(pressedRow, pressedCol, e.isControlDown(), e.isShiftDown());
*/
@Override
public void changeSelection(int rowIndex, int columnIndex, boolean toggle, boolean extend)
{
if (toggle && !isRowSelected(rowIndex))
return; // Don't do the selection
super.changeSelection(rowIndex, columnIndex, toggle, extend);
}
}
Гораздо проще и именно то, что вам нужно!
Кстати, спасибо за предоставление такого простого хорошего тестового примера, я, возможно, не попробовал бы, если бы мне пришлось написать его самому ... :-D Это была интересная и обучающая задача.
Другие советы
У меня был успех со следующим, хотя я не уверен, что это лучший метод ...
class SpecialTable extends JTable
{
boolean bIsControlDown;
int clickedRow;
SpecialTable(Object[][] data, String[] columnNames)
{
super(data, columnNames);
// setSelectionMode(ListSelectionModel.MULTIPLE_INTERVAL_SELECTION);
getSelectionModel().addListSelectionListener(this);
addMouseListener(new MouseInputAdapter()
{
public void mousePressed(MouseEvent me)
{
bIsControlDown = me.isControlDown();
clickedRow = rowAtPoint(me.getPoint());
}
});
}
public void valueChanged(ListSelectionEvent evt)
{
super.valueChanged(evt);
if (bIsControlDown)
{
if (!evt.getValueIsAdjusting())
{
// System.out.println(evt);
// System.out.println("=> " + clickedRow);
getSelectionModel().removeSelectionInterval(clickedRow, clickedRow);
}
}
}
}
Замените строки, определяющие table
в вашем коде, только на:
JTable table = new SpecialTable(data, columnNames);
table.setPreferredScrollableViewportSize(new Dimension(500, 100));
Когда вы удерживаете нажатой клавишу управления невыбранной строкой, она кратко выбирается, а затем не выбирается.