Introducción
Una vez leí la siguiente cita:
Any sufficiently advanced bug is indistinguishable from a feature.
- Rich Kulawiec
(Cualquier fallo suficientemente sofisticado es indistinguible de una característica).
En la documentación de Java Swing se dice que el componente JTextField no acepta varias líneas y que para ese caso hay otras opciones como JTextArea o JTextPane.
Efectivamente, si se pulsa la tecla ENTER cuando el "focus" lo tiene un JTextField no se produce un salto de línea. Pero, ¿qué ocurre si se establece el texto del JTextField mediante el método JTextField#setText(String) con una String que sí contiene saltos de línea?
Pues he descubierto que depende, tal y como se explica en este "fallo documentado":
JDK-6427290: Possibility to put newlines into a JTextField
Esto ocurre al menos en la versiones 1.6.0_37, 1.7.0_25 y anteriores.
Ummm, ¿es un fallo o es una característica útil? Eso lo dejo a tu criterio.
Demostración
La clave está en instalar sobre el JTextField un DocumentFilter que no altere los saltos de línea presentes en la String (en realidad se instala sobre el Document del JTextField).
Por defecto, los JTextField no tienen ningún DocumentFilter instalado y en tal caso el propio JTextField sustituye los saltos de línea que detecta en la String por espacios en blanco (según JDK-6427290).
Sin embargo, al instalar un DocumentFilter, el JTextField no altera la String y delega completamente esa responsabilidad al DocumentFilter.
El pequeño programa JTextFieldLinesTest crea una ventana con un JTextArea, un JTextField y un JToggleButton:
- El TextArea muestra información del sistema.
- El TextField sirve para hacer pruebas con saltos de línea.
- El botón permite conmutar el comportamiento del TextField respecto a los saltos de línea.
Puedes ver y copiar el código fuente completo en el siguiente enlace:
Las líneas que más interesan del programa están en el ItemListener del botón. Según se invoque a setDocumentFilter(DocumentFilter) con null o con un DocumentFilter válido variará el comportamiento del JTextField respecto a los saltos de línea.
Document document= textField.getDocument(); AbstractDocument abstractDocument= (AbstractDocument) document; DocumentFilter filter; if (button.isSelected() == true) { filter= new DocumentFilter(); } else { filter= null; } abstractDocument.setDocumentFilter(filter);
En estas capturas se puede ver el como cambia el comportamiento y el aspecto visual del JTextField.
Monolínea |
Multilínea |
En ningún caso se pueden teclear saltos de línea (tecla ENTER) sobre el JTextField, pero sí se pueden pegar desde el portapapeles textos que contengan saltos de línea.
Respecto al aspecto visual del JTextField, el que se estire o no a lo alto por la presencia de saltos de línea depende de diversos factores como el LayoutManager que se esté utilizando. Si se muestran saltos de línea sin estirar el JTextField a lo alto el contenido se verá horrible.
Comentar a este respecto que con los tabuladores pasa algo parecido. Al pulsar la tecla TAB normalmente se cede el "focus" al siguiente componente, por lo que no se pueden teclear tabuladores. Sin embargo, si la String indicada en JTextField#setText(String) o el texto pegado desde el portapapeles contienen tabuladores el JTextField los mostrará.
Conclusiones
En mi humilde opinión, mientras no se cambie este comportamiento, es una característica que resulta muy útil para detectar visualmente Strings que contengan saltos de línea en JTextField que se rellenen programáticamente, por ejemplo, con datos leídos desde una base de datos que supuestamente no deberían contener saltos de línea (por eso se visualizan con un JTextField).
Enlaces
Enlaces de interés relacionados con este artículo:
- JDK-6427290: Possibility to put newlines into a JTextField
- JTextField
- DocumentFilter
- Document
- AbstractDocument
(Actualizado 27/11/2013)