Вкладка клавиш в JavaFX TextArea
Как мне попасть в клавишу Tab в TextArea, перейдет к следующему элементу управления?
Я мог бы добавить слушателя к событию с нажатой клавишей, но как заставить элемент управления TextArea потерять фокус (не зная, что следующее поле цепочки будет сфокусировано)?
@FXML protected void handleTabKeyTextArea(KeyEvent event) {
if (event.getCode() == KeyCode.TAB) {
...
}
}
Ответы
Ответ 1
Этот код пересекает фокус, если нажать TAB и вставить вкладку, если нажать CONTROL + TAB
textArea.addEventFilter(KeyEvent.KEY_PRESSED, new EventHandler<KeyEvent>() {
@Override
public void handle(KeyEvent event) {
if (event.getCode() == KeyCode.TAB) {
SkinBase skin = (SkinBase) textArea.getSkin();
if (skin.getBehavior() instanceof TextAreaBehavior) {
TextAreaBehavior behavior = (TextAreaBehavior) skin.getBehavior();
if (event.isControlDown()) {
behavior.callAction("InsertTab");
} else {
behavior.callAction("TraverseNext");
}
event.consume();
}
}
}
});
Ответ 2
Я использую методы трассировки
@Override
public void handle(KeyEvent event) {
if (event.getCode().equals(KeyCode.TAB)) {
Node node = (Node) event.getSource();
if (node instanceof TextField) {
TextFieldSkin skin = (TextFieldSkin) ((TextField)node).getSkin();
if (event.isShiftDown()) {
skin.getBehavior().traversePrevious();
}
else {
skin.getBehavior().traverseNext();
}
}
else if (node instanceof TextArea) {
TextAreaSkin skin = (TextAreaSkin) ((TextArea)node).getSkin();
if (event.isShiftDown()) {
skin.getBehavior().traversePrevious();
}
else {
skin.getBehavior().traverseNext();
}
}
event.consume();
}
}
Ответ 3
Начиная с Java 9 (2017), большинство ответов на этой странице не работают, поскольку вы больше не можете делать skin.getBehavior()
.
Это работает:
@Override
public void handle(KeyEvent event) {
KeyCode code = event.getCode();
if (code == KeyCode.TAB && !event.isShiftDown() && !event.isControlDown()) {
event.consume();
Node node = (Node) event.getSource();
try {
Robot robot = new Robot();
robot.keyPress(KeyCode.CONTROL.getCode());
robot.keyPress(KeyCode.TAB.getCode());
robot.delay(10);
robot.keyRelease(KeyCode.TAB.getCode());
robot.keyRelease(KeyCode.CONTROL.getCode());
}
catch (AWTException e) { }
}
}
Это также работает:
@Override
public void handle(KeyEvent event) {
KeyCode code = event.getCode();
if (code == KeyCode.TAB && !event.isShiftDown() && !event.isControlDown()) {
event.consume();
Node node = (Node) event.getSource();
KeyEvent newEvent
= new KeyEvent(event.getSource(),
event.getTarget(), event.getEventType(),
event.getCharacter(), event.getText(),
event.getCode(), event.isShiftDown(),
true, event.isAltDown(),
event.isMetaDown());
node.fireEvent(newEvent);
}
}
Оба имитируют нажатие CTRL+TAB
, когда пользователь нажимает TAB
. Поведение TextArea по умолчанию для CTRL+TAB
по умолчанию перемещает фокус на следующий элемент управления. Обратите внимание, что второй код основан на ответе Йохана Де Шуттера.
Ответ 4
Если другое решение проблемы Tab-Focus.
Поведение TextArea по умолчанию для клавиши CTRL + TAB по умолчанию - это переход фокуса к следующему элементу управления. Поэтому я заменил событие ключа TAB ключевым событием CTRL + TAB, и когда пользователь нажимает CTRL + TAB, в TextArea вставлен символ табуляции.
Мой вопрос: нормально ли запускать событие в фильтре событий? И нормально ли заменить текст KeyEvent на FOCUS_EVENT_TEXT, чтобы иметь указание, является ли это событием, созданным пользователем, или событием, созданным в фильтре событий.
Фильтр событий:
javafx.scene.control.TextArea textArea1 = new javafx.scene.control.TextArea();
textArea1.addEventFilter(KeyEvent.KEY_PRESSED, new TextAreaTabToFocusEventHandler());
Обработчик событий:
public class TextAreaTabToFocusEventHandler implements EventHandler<KeyEvent>
{
private static final String FOCUS_EVENT_TEXT = "TAB_TO_FOCUS_EVENT";
@Override
public void handle(final KeyEvent event)
{
if (!KeyCode.TAB.equals(event.getCode()))
{
return;
}
// handle events where the TAB key or TAB + CTRL key is pressed
// so don't handle the event if the ALT, SHIFT or any other modifier key is pressed
if (event.isAltDown() || event.isMetaDown() || event.isShiftDown())
{
return;
}
if (!(event.getSource() instanceof TextArea))
{
return;
}
final TextArea textArea = (TextArea) event.getSource();
if (event.isControlDown())
{
// if the event text contains the special focus event text
// => do not consume the event, and let the default behaviour (= move focus to the next control) happen.
//
// if the focus event text is not present, then the user has pressed CTRL + TAB key,
// then consume the event and insert or replace selection with tab character
if (!FOCUS_EVENT_TEXT.equalsIgnoreCase(event.getText()))
{
event.consume();
textArea.replaceSelection("\t");
}
}
else
{
// The default behaviour of the TextArea for the CTRL+TAB key is a move of focus to the next control.
// So we consume the TAB key event, and fire a new event with the CTRL + TAB key.
event.consume();
final KeyEvent tabControlEvent = new KeyEvent(event.getSource(), event.getTarget(), event.getEventType(), event.getCharacter(),
FOCUS_EVENT_TEXT, event.getCode(), event.isShiftDown(), true, event.isAltDown(), event.isMetaDown());
textArea.fireEvent(tabControlEvent);
}
}
}
Ответ 5
Вдохновленный предыдущими ответами и для очень похожего случая, я построил следующий класс:
/**
* Handles tab/shift-tab keystrokes to navigate to other fields,
* ctrl-tab to insert a tab character in the text area.
*/
public class TabTraversalEventHandler implements EventHandler<KeyEvent> {
@Override
public void handle(KeyEvent event) {
if (event.getCode().equals(KeyCode.TAB)) {
Node node = (Node) event.getSource();
if (node instanceof TextArea) {
TextAreaSkin skin = (TextAreaSkin) ((TextArea)node).getSkin();
if (!event.isControlDown()) {
// Tab or shift-tab => navigational action
if (event.isShiftDown()) {
skin.getBehavior().traversePrevious();
} else {
skin.getBehavior().traverseNext();
}
} else {
// Ctrl-Tab => insert a tab character in the text area
TextArea textArea = (TextArea) node;
textArea.replaceSelection("\t");
}
event.consume();
}
}
}
}
Я просто не видел необходимости обрабатывать вкладку в контексте TextField, поэтому я удалил эту часть.
Затем этот класс можно легко использовать, как описано User:
TextArea myTextArea = new TextArea();
mytTextArea.addEventFilter(KeyEvent.KEY_PRESSED, new TabTraversalEventHandler());
И все это работает как шарм:)
Ответ 6
У меня была такая же проблема, и мне нравятся методы трассировки, которые использует Том.
Но я также хочу вставить вкладку, когда нажата клавиша Ctrl +.
Вызов
behavior.callAction("InsertTab");
не работает с JavaFX8. Взгляд в классе TextAreaBehaviour показал мне, что теперь есть действие "TraverseOrInsertTab".
Но, однако, я думаю, что такое действие вызывает довольно неустойчивое в нескольких версиях Java, потому что оно полагается на переданную строку.
Поэтому вместо метода callAction() я использовал
textArea.replaceSelection("\t");