2009-03-12 14 views
5

Czy istnieje łatwy sposób manipulowania elementami sterującymi na JTable, aby nadać inną funkcjonalność po naciśnięciu przycisku klawiatury (np. Przycisku CTRL) i wybraniu wiersza? Poproszono mnie o utworzenie tabeli, w której CTRL + kliknięcie (kliknięcie myszką) w rzędzie odkaże tylko zaznaczony wiersz, nigdy nie wybierze wiersza. Jeśli użytkownik kliknie + kliknie w niezaznaczony wiersz, nic się nie stanie.Zmiana zachowania Ctrl + kliknięcie na JTable

Udało mi się utworzyć tabelę i wyłączyć funkcje, takie jak CTRL + A (zaznacz wszystko), i mogłem sprawdzić, czy przycisk kontrolny jest naciśnięty podczas generowania zdarzenia MouseEvent, ale mogę " Wydaje się, że można wyregulować CTRL + Click. Oto niektóre kodu:

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

W metodzie mousePressed Grałem około z coraz wszystkie wybrane wiersze z tabeli, a następnie zamierza sprawdzić, czy nowo kliknięciu rząd był w SelectedRows ... Jednak Nie jestem pewien, czy istnieje sposób, aby zobaczyć, który wiersz jest powiązany z MouseEvent.

(Również wiem oczekiwanego zachowania, takie jak ten nie powinien być bawił się zbyt wiele, ale to replikować starszego systemu w firmie)

Wszelkie pomysły/sugestie będą mile widziane!

Odpowiedz

8

OK, drugi raz (opuściłem pierwszy, ponieważ może zainteresować kogoś do jakiegoś innego użycia, kto wie? Powiedz, że jest tam dla celów edukacyjnych ... :-)).

Spojrzałem na kod źródłowy JTable i odkryłem, że zdarzenia myszy są obsługiwane przez wygląd i styl. Wiedząc, jak obsługuje klucz kontrolny, mogłem bezpiecznie zastąpić metodę changeSelection, aby zrobić to, czego potrzebujesz.
Uważam, że wymagania są nieco dziwne (nadal możesz używać Shift + kliknięcie, nie?), Ale nie znam kontekstu.

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

Znacznie prostsze i dokładnie to, czego potrzebujesz!

BTW, dzięki za dostarczenie tak prostego, dobrego testu, mógłbym nie spróbować, gdybym musiał sam to napisać ... :-D To było ciekawe i uczące wyzwanie.

+0

Woohoo! Zadziałało! Dzięki :) I tak, myślę, że wymagania są szalone ... a to nawet nie połowa! – Nicks

1

miałem sukces z następującymi, choć nie jestem pewien, że to najlepszy sposób ...

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

Wymień linie wyznaczające table w kodzie tylko z:

JTable table = new SpecialTable(data, columnNames); 
    table.setPreferredScrollableViewportSize(new Dimension(500, 100)); 

kiedy przytrzymując klawisz Ctrl w niezaznaczonym wierszu, zostanie krótko zaznaczony, a następnie odznaczony.

+0

Dzięki za to. Niestety, będzie to działało na mój problem. Zaraz po wybraniu wiersza następuje odpalenie komunikatu, więc czynność szybkiego wyboru i anulowania nadal będzie wysyłała ten komunikat. – Nicks

+0

Obawiałem się tego. Alternatywą jest uniknięcie wywołania super.valueChanged() i w pełni obsłużyć sam wybór wielokrotnego przedziału (może nawet skopiować kod Java i zmienić go na swój sposób, jak sądzę). – PhiLho

+0

Prawda - dzięki, myślę, że dam to odejść – Nicks

Powiązane problemy