PDA

View Full Version : [Java] non far selezionare una cella su Jtable


spv42
19-01-2007, 11:11
Ciao,

Vorrei impostare la mia tabella in modo che possa decidere quali celle (o colonne) sian selezionabili o meno.

E' possibile?

:help:

Grazie

RaouL_BennetH
19-01-2007, 14:56
Dalla documentazione leggo che c'è il metodo:

isCellEditable(int row, int column);


magari vedi se trovi qualcosa di utile in questo link:

http://forum.java.sun.com/thread.jspa?threadID=384040&messageID=1648959

spv42
22-01-2007, 09:24
Dalla documentazione leggo che c'è il metodo:

isCellEditable(int row, int column);


Io però intendevo non selezionabile, quel metodo lo conosco, non permette di editare la cella, ma la stessa cella risulta selezionabile!

Io vorrei che neanche mi si "accendesse" :muro:

RaouL_BennetH
22-01-2007, 15:51
Io però intendevo non selezionabile, quel metodo lo conosco, non permette di editare la cella, ma la stessa cella risulta selezionabile!

Io vorrei che neanche mi si "accendesse" :muro:

uhm... ma un "setEnabled(false)" per le celle che ti interessa non rendere selezionabili?

spv42
22-01-2007, 17:06
uhm... ma un "setEnabled(false)" per le celle che ti interessa non rendere selezionabili?

Quel metodo si applica a tutto il componente JTable e non alla singola cella!
Il problema è proprio questo!
Sembrerebbe che le celle non sono viste come oggetti!!! :muro:

:help:

spv42
23-01-2007, 10:16
Nessuno può indicarmi la retta via?!?!? :mc: :muro:

PGI-Bis
23-01-2007, 15:43
Puoi sovrascrivere il metodo isCellSelected di JTable. Se la cella non deve essere selezionata restituisci false. Dovresti anche creare un TableCellRenderer per togliere il focus alla cella nel caso in cui non sia selezionabile.

spv42
24-01-2007, 19:10
Come lo tolgo il focus?!?!

Posso installare sul TableCellRenderer un FocusListner, ma i metodi implementabili sono solo FocusGained e FocusLost.


:help: me per Favore :muro:

PGI-Bis
24-01-2007, 22:35
Tra i parametri del metodo getTableCellRendererComponent c'è un boolean, isFocused. Puoi sovrascrivere il metodo in questione di DefaultTableCellRenderer in modo tale da rendere false quel parametro, prima di invocare la super definizione del metodo, se gli indici di riga e colonna corrispondano a quelli di una cella non selezionabile.

spv42
25-01-2007, 10:59
Tra i parametri del metodo getTableCellRendererComponent c'è un boolean, isFocused. Puoi sovrascrivere il metodo in questione di DefaultTableCellRenderer in modo tale da rendere false quel parametro, prima di invocare la super definizione del metodo, se gli indici di riga e colonna corrispondano a quelli di una cella non selezionabile.

Nei metodi di DefaultTableCellRenderer non c'è!!!

Dov'è?

PGI-Bis
25-01-2007, 13:30
Uomo di poca fede, se dico che c'è, c'è :D. Faccio un esempio:

import javax.swing.*;
import javax.swing.table.*;
import java.awt.*;

public class CustomCellRenderer extends DefaultTableCellRenderer {
private SelectionData selectionData;

public CustomCellRenderer(SelectionData data) {
this.selectionData = data;
}

public Component getTableCellRendererComponent(JTable table,
Object value, boolean isSelected, boolean hasFocus,
int row, int column)
{
if(hasFocus && !selectionData.canSelect(row, column)) {
hasFocus = false;
}
return super.getTableCellRendererComponent(table,
value, isSelected, hasFocus, row, column);
}
}

Qui SelectionData è un tipo di oggetti creato ad hoc per mantenere un elenco di celle non selezionabili, individuate dagli indici di riga e colonna.

spv42
27-01-2007, 16:12
Il mio problema è che ho un mio CellRenderer Che implementa TableCellRenderer!

Ho provato a impostare il boolean hasFocus a false, la cella non mi si seleziona più, ma il focus lo acquisisce lo stesso, o almeno, quando premo tab per saltare dalla cella che la precede a quella successiva devo premere il tasto 2 volte!!!

:muro: :muro: :muro: :muro:

spv42
29-01-2007, 14:05
Uomo di poca fede, se dico che c'è, c'è :D. Faccio un esempio:

import javax.swing.*;
import javax.swing.table.*;
import java.awt.*;

public class CustomCellRenderer extends DefaultTableCellRenderer {
private SelectionData selectionData;

public CustomCellRenderer(SelectionData data) {
this.selectionData = data;
}

public Component getTableCellRendererComponent(JTable table,
Object value, boolean isSelected, boolean hasFocus,
int row, int column)
{
if(hasFocus && !selectionData.canSelect(row, column)) {
hasFocus = false;
}
return super.getTableCellRendererComponent(table,
value, isSelected, hasFocus, row, column);
}
}

Qui SelectionData è un tipo di oggetti creato ad hoc per mantenere un elenco di celle non selezionabili, individuate dagli indici di riga e colonna.


Ho provato per scrupolo, ma non funziona!
Il Focus lo acquisisce lo stesso!!!!

:help:

PGI-Bis
29-01-2007, 18:01
Scrivo un esempio completo. Il problema generale è che una tabella supporta una varietà di modelli di selezione e il caso peggiore è certamente quello in cui tutti questi modelli siano desiderati. Vale a dire che è più facile trattare l'esclusione di una cella dalla selezione qualora il modello di selezione preveda esclusivamente l'accesso ad una singola cella per volta. JTable, al contrario, autorizza normalmente la selezione di più colonne ed eventualmente di una singola cella all'interno di una di queste. Detto questo, l'esempio parte con un oggetto in grado di stabilire quali singole celle all'interno della tabella non siano selezionabili.

import java.awt.Point;
import java.util.*;

public class SelectionData {
private ArrayList<Point> unselectableCells = new ArrayList<Point>();

public SelectionData(int...cellCoords) {
for(int i = 0; i < cellCoords.length; i += 2) {
unselectableCells.add(new Point(cellCoords[i], cellCoords[i + 1]));
}
}

public boolean canSelectCell(int row, int col) {
Point p = new Point(row, col);
return unselectableCells.contains(p) == false;
}
}

Volendo istruire SelectionData ad escludere le celle (0, 0) e (3, 4), lo si creerà con:

SelectionData sd = new SelectionData(0, 0, 3, 4);

Non ho trovato un nome migliore di SelectionData :D.

Il resto si appoggia a questo strumento. Il primo di questo resto è il proiettore delle celle. Il proiettore si rende necessario qualora il modello di selezione della tabella consenta di scegliere intere righe o colonne. Il proiettore fa sì che una cella abbia l'aspetto "non selezionato" quando, date le sue coordinate, essa risulti non selezionabile.

import java.awt.*;
import javax.swing.*;
import javax.swing.table.*;

public class MyRenderer extends DefaultTableCellRenderer {
private SelectionData selectionData;

public MyRenderer(SelectionData selectionData) {
this.selectionData = selectionData;
}

public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column) {
if(selectionData.canSelectCell(row, column) == false) {
isSelected = false;
hasFocus = false;
}
return super.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
}
}

Il secondo componente che si appoggia a SelectionData è una tabella personalizzata. La personalizzazione sta in ciò che la tabella esclude la possibilità di mutare il valore di una cella non selezionabile oltre ad indicare come non selezionata una cella se ciò sia escluso da SelectionData.

import javax.swing.*;

public class MyTable extends JTable {
private SelectionData selectionData;

public MyTable(SelectionData selectionData) {
this.selectionData = selectionData;
}

public boolean isCellSelected(int row, int col) {
boolean answer = super.isCellSelected(row, col);
return answer ? selectionData.canSelectCell(row, col) : false;
}

public boolean isCellEditable(int row, int col) {
boolean answer = super.isCellEditable(row, col);
return answer ? selectionData.canSelectCell(row, col) : false;
}
}

Il terzo incomodo è l'oggetto che gestisce la mutazione di selezione in corrispondenza della pressione dei tasti di controllo. L'esempio contiene unicamente il trattamento del tasto "tab" ma è facile estenderlo anche alle frecce. E' un KeyListener. Gli eventi di pressione di un tasto sono propagati dalla tabella prima che essi si riflettano sui suoi due ListSelectionModel.

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;

public class SelectionKeyManager extends KeyAdapter {
private SelectionData selectionData;
private JTable table;
private ListSelectionModel rowSelectionModel, colSelectionModel;

public SelectionKeyManager(SelectionData selectionData, JTable table) {
this.selectionData = selectionData;
this.table = table;
rowSelectionModel = table.getSelectionModel();
colSelectionModel = table.getColumnModel().getSelectionModel();
}

public void keyPressed(KeyEvent e) {
if(e.getKeyCode() == KeyEvent.VK_TAB) {
if((e.getModifiersEx() & InputEvent.SHIFT_DOWN_MASK) == InputEvent.SHIFT_DOWN_MASK) {
previousColumn();
} else {
nextColumn();
}
} else if(e.getKeyCode() == KeyEvent.VK_UP) {

} else if(e.getKeyCode() == KeyEvent.VK_LEFT) {

} else if(e.getKeyCode() == KeyEvent.VK_RIGHT) {

} else if(e.getKeyCode() == KeyEvent.VK_DOWN) {

}
}

private void previousColumn() {
int row = rowSelectionModel.getLeadSelectionIndex();
int col = colSelectionModel.getLeadSelectionIndex() - 1;
if(row < 0) {
row = table.getRowCount() - 1;
}
if(col < 0) {
col = table.getColumnCount() - 1;
}
boolean canSelect = selectionData.canSelectCell(row, col);
if(!canSelect) {
colSelectionModel.setLeadSelectionIndex(col);
}
}

private void nextColumn() {
int row = rowSelectionModel.getLeadSelectionIndex();
int col = colSelectionModel.getLeadSelectionIndex() + 1; //key press invocato prima della mutazione di selezione
if(row < 0) {
row = 0;
}
if(col == table.getColumnCount()) {
row++;
if(row == table.getRowCount()) {
row = 0;
rowSelectionModel.setSelectionInterval(0, 0);
}
col = 0;
}
boolean canSelect = selectionData.canSelectCell(row, col);
if(!canSelect) {
colSelectionModel.setLeadSelectionIndex(col);
}
}

private void cellUp() {

}

private void cellDown() {

}

private void cellLeft() {

}

private void cellRight() {

}
}

La classe Main che segue mette insieme il tutto.

import java.awt.*;
import javax.swing.*;
import javax.swing.table.*;

public class Main implements Runnable {
public static void main(String[] args) {
SwingUtilities.invokeLater(new Main());
}

public void run() {
DefaultTableModel tableModel = new DefaultTableModel(5, 10);
SelectionData selectionData = new SelectionData(0, 0, 1, 1, 2, 2, 3, 3);
MyTable table = new MyTable(selectionData);
table.setModel(tableModel);
MyRenderer renderer = new MyRenderer(selectionData);
table.setDefaultRenderer(Object.class, renderer);
SelectionKeyManager selectionKeyManager = new SelectionKeyManager(selectionData, table);
table.addKeyListener(selectionKeyManager);
JScrollPane scrollPane = new JScrollPane(table);
JFrame window = new JFrame("Table sample");
window.add(scrollPane);
window.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
window.pack();
window.setVisible(true);
}
}

Ci sono delle alternative. Puoi definire un tuo ListSelectionModel (ne servono due, uno per le righe e uno per le colonne). Personalmente ritengo più semplice intervenire "dall'esterno", manipolando l'input, anzichè dall'interno, sostituendo in toto il modello di selezione della tabella. La questione della semplicità è tuttavia sempre personale e opinabile.