Вопрос

Есть ли простой способ манипулировать элементами управления в 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));

Когда вы удерживаете нажатой клавишу управления невыбранной строкой, она кратко выбирается, а затем не выбирается.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top