sábado, 1 de marzo de 2014

Adaptando el JList

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

No hay comentarios:

Publicar un comentario