2009-03-12 3 views
5

Есть ли простой способ управления элементами управления на JTable, чтобы придать клавиатуру нажатой клавишей CTRL (например, кнопку CTRL) и выбрать строку? Мне было предложено создать таблицу, в которой CTRL + Click (щелчок мышью) в строке будет отменять только выбранную строку, никогда не выбирать строку. Если пользователь CTRL + щелкнет невыбранную строку, ничего не произойдет.Изменение поведения Ctrl + click на JTable

Я смог создать таблицу и отключить функции, такие как CTRL + A (выбрать все), и я смог проверить, нажата ли кнопка управления при создании MouseEvent, но я могу " t, похоже, выяснить, как можно скопировать 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.

(Кроме того, я знаю, что ожидаемое поведение, такое как это не должно играть вокруг слишком много, но это повторить унаследованную систему в компании)

Любые идеи/предложения будут оценены!

ответ

8

ОК, второй раз (я оставил первый, так как это может заинтересовать кого-то для другого использования, кто знает? Скажите, что он есть в образовательной цели ... :-)).

Я посмотрел исходный код JTable и узнал, что события мыши обрабатываются внешним видом. Зная, как он управляет ключом управления, я мог бы безопасно переопределить метод changeSelection, чтобы делать то, что вам нужно.
Я нахожу требования немного странными (вы все равно можете использовать Shift + click, no?), Но я не знаю контекста.

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); 
    } 
} 

Гораздо проще и точно, что вам нужно!

BTW, спасибо за предоставление такого простого теста, я, возможно, не пробовал, если бы мне пришлось написать его сам ... :-D Это был интересный и обучающий вызов.

+0

Woohoo! Это сработало! Спасибо :) И да, я думаю, что требования сумасшедшие ... и это даже не половина его! – Nicks

1

Я имел успех с нижеследующим, хотя я не уверен, что это лучший способ ...

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)); 

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

+0

Спасибо за это. К сожалению, это будет работать для моей проблемы. Как только строка будет выбрана, она отключит сообщение, поэтому быстрое действие select-deselect по-прежнему отправит это сообщение. – Nicks

+0

Я боялся этого. Альтернатива заключается в том, чтобы избежать вызова super.valueChanged() и полностью обрабатывать выбор нескольких интервалов самостоятельно (могу даже скопировать код Java и изменить его по-своему, я думаю). – PhiLho

+0

Правда - спасибо, я думаю, что я дам это зайти – Nicks