Para el Yoching me he encontrado con la necesidad de adaptar el sistema de selección de un
JList
. El comportamiento estándar de una lista es que si se clickea sobre un elemento, éste pasa a estar seleccionado, borrando la selección anterior a no ser que se mantengan pulsadas las teclas shift, control o command, haciendo entonces selecciones múltiples, continúas o discontinúas.
Lo que necesitaba era que la lista simulara el comportamiento de una sucesión de checkboxes, es decir, que si pulso sobre un elemento de la lista, este elemento permuta su estado de selección, manteniendo la selección del resto de la lista inalterada.
Hay varias soluciones para esto, cada cual que elija la suya. Se puede colocar una serie de
JCheckBox
en un
JPanel
con layout vertical. Se puede hacer un
ListCellRenderer
que herede de un
JCheckBox
e implementar un controlador de activaciones. En cambio, he atacado el problema manipulando el modelo de selección del
JList
: he creado mi propio
ListSelectionModel
adaptado la clase
DefaultListSelectionModel
.
Partiendo del código fuente de
DefaultListSelectionModel
, encuentro que hay varios métodos que necesito retocar para tener el modo de selección buscado. Por desgracia, para reescribirlos no puedo heredar mi clase de
DefaultListSelectionModel
, pues en sus tripas se hacen referencia a métodos y campos privados, y lo más fácil es reescribir la clase fuente.
Vamos al código
El primer método en cuestión es
setSelectionInterval
. Como su nombre indica, establece el intervalo de selección, reseteando el estado anterior. Lo que hay que hacer es que llame directamente a
addSelectionInterval
, es decir, que añada el nuevo intervalo a la selección existente:
public void setSelectionInterval(int index0, int index1) {
addSelectionInterval(index0, index1);
}
En
addSelectionInterval
obviamos las opciones que no nos interesa, pues nuestro modo de edición es
MULTIPLE_INTERVAL_SELECTION
, y así llegamos al método privado
changeSelection
. Simplificamos mucho la función original, eliminando el parámetro booleano
clearFirst
e intercambiando el estado de los items de la nueva selección:
private void changeSelection(int clearMin, int clearMax, int setMin, int setMax) {
for (int i = Math.min(setMin, clearMin); i <= Math.max(setMax, clearMax); i++) {
flip(i);
}
fireValueChanged();
}
private void flip(int r) {
if (value.get(r))
clear(r);
else
set(r);
}
Y que no se nos olvide mantener el modo de selección fijo:
public void setSelectionMode(int selectionMode) {
this.selectionMode = MULTIPLE_INTERVAL_SELECTION;
}
¡Ya sólo falta decirle al
JList
que queremos tener este modelo de selección!
JList list = new JList();
list.setSelectionModel(new AccumulativeSelectionModel());