2009-03-12 13 views
5

Esiste un modo semplice per manipolare i controlli su una tabella J per fornire funzionalità diverse quando viene premuto un pulsante della tastiera (ad esempio il pulsante CTRL) e viene selezionata una riga? Mi è stato chiesto di creare una tabella in cui CTRL + clic (clic del mouse) su una riga deselezionerà solo una riga selezionata, mai selezionare una riga. Se l'utente CTRL + fa clic su una riga non selezionata, non accadrà nulla.Cambia il comportamento di Ctrl + clic su una JTable

Sono stato in grado di creare una tabella e disattivare funzioni come CTRL + A (seleziona tutto) e sono stato in grado di controllare se il pulsante di controllo viene premuto quando viene generato un mouseEvent, ma posso ' t sembra capire come il CTRL + clic può essere regolato. Ecco po 'di codice:

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

Nel metodo mousePressed Ho giocato in giro con ottenere tutte le righe selezionate dalla tabella, e stavo andando poi a verificare se la riga appena cliccato era negli selectedRows ... Tuttavia, Non sono sicuro se c'è un modo per vedere quale riga è associata a MouseEvent.

(Inoltre, so comportamento previsto come questo non dovrebbe essere giocato con troppo, ma è di replicare un sistema legacy in azienda)

Qualsiasi idee/suggerimenti sarebbero apprezzati!

risposta

8

OK, seconda ripresa (ho lasciato il primo perché potrebbe interessare qualcuno per qualche altro utilizzo, chi lo sa? Diciamo che è lì per scopi didattici ... :-)).

Ho dato un'occhiata al codice sorgente di JTable e ho scoperto che gli eventi del mouse sono gestiti dall'aspetto. Sapendo come gestisce la chiave di controllo, posso tranquillamente ignorare il metodo changeSelection per fare ciò che ti serve.
Trovo che i requisiti siano un po 'strani (puoi ancora usare MAIUSC + clic, no?) Ma non conosco il contesto.

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

Molto più semplice ed esattamente quello che ti serve!

BTW, grazie per aver fornito un così semplice test case, non avrei potuto provare se dovessi scrivere da solo ... MrGreen È stata una sfida interessante e di apprendimento.

+0

Woohoo! Ha funzionato! Grazie :) E sì, penso che le esigenze siano pazzesche ... e non è nemmeno la metà! – Nicks

1

ho avuto successo con il seguente, anche se non sono sicuro che è il metodo migliore ...

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

Sostituire le linee che definiscono table nel codice con solo:

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

Quando si control-clic su una riga non selezionata, viene brevemente selezionata, quindi deselezionata.

+0

Grazie per quello. Sfortunatamente funzionerà comunque per il mio problema. Non appena viene selezionata una riga, spara il messaggio, quindi l'azione di selezione rapida-deseleziona continuerà a inviare quel messaggio. – Nicks

+0

Lo temevo. Un'alternativa è quella di evitare la chiamata a super.valueChanged() e gestire completamente la selezione a intervalli multipli (può anche copiare il codice di Java e modificarlo a modo tuo, penso). – PhiLho

+0

Vero - grazie, penso che lo farò andare – Nicks

Problemi correlati