/* * Copyright (c) 2011, 2014, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package com.sun.javafx.scene.control.behavior; import com.sun.javafx.scene.control.inputmap.InputMap; import javafx.event.Event; import javafx.scene.control.SelectionModel; import javafx.scene.control.Tab; import javafx.scene.control.TabPane; import javafx.scene.input.*; import com.sun.javafx.scene.control.inputmap.KeyBinding; import java.util.List; import static javafx.scene.input.KeyCode.*; import static com.sun.javafx.scene.control.inputmap.InputMap.KeyMapping; import static com.sun.javafx.scene.control.inputmap.InputMap.MouseMapping; public class TabPaneBehavior extends BehaviorBase { private final InputMap tabPaneInputMap; public TabPaneBehavior(TabPane tabPane) { super(tabPane); // create a map for TabPane-specific mappings (this reuses the default // InputMap installed on the control, if it is non-null, allowing us to pick up any user-specified mappings) tabPaneInputMap = createInputMap(); // TabPane-specific mappings for key and mouse input addDefaultMapping(tabPaneInputMap, new KeyMapping(UP, e -> selectPreviousTab()), new KeyMapping(DOWN, e -> selectNextTab()), new KeyMapping(LEFT, e -> rtl(tabPane, this::selectNextTab, this::selectPreviousTab)), new KeyMapping(RIGHT, e -> rtl(tabPane, this::selectPreviousTab, this::selectNextTab)), new KeyMapping(HOME, e -> { if (getNode().isFocused()) { moveSelection(0, 1); } }), new KeyMapping(END, e -> { if (getNode().isFocused()) { moveSelection(getNode().getTabs().size() - 1, -1); } }), new KeyMapping(new KeyBinding(PAGE_UP).ctrl(), e -> selectPreviousTab()), new KeyMapping(new KeyBinding(PAGE_DOWN).ctrl(), e -> selectNextTab()), new KeyMapping(new KeyBinding(TAB).ctrl(), e -> selectNextTab()), new KeyMapping(new KeyBinding(TAB).ctrl().shift(), e -> selectPreviousTab()), new MouseMapping(MouseEvent.MOUSE_PRESSED, e -> getNode().requestFocus()) ); } @Override public InputMap getInputMap() { return tabPaneInputMap; } public void selectTab(Tab tab) { getNode().getSelectionModel().select(tab); } public boolean canCloseTab(Tab tab) { Event event = new Event(tab,tab,Tab.TAB_CLOSE_REQUEST_EVENT); Event.fireEvent(tab, event); return ! event.isConsumed(); } public void closeTab(Tab tab) { TabPane tabPane = getNode(); // only switch to another tab if the selected tab is the one we're closing int index = tabPane.getTabs().indexOf(tab); if (index != -1) { tabPane.getTabs().remove(index); } if (tab.getOnClosed() != null) { Event.fireEvent(tab, new Event(Tab.CLOSED_EVENT)); } } // Find a tab after the currently selected that is not disabled. Loop around // if no tabs are found after currently selected tab. public void selectNextTab() { moveSelection(1); } // Find a tab before the currently selected that is not disabled. public void selectPreviousTab() { moveSelection(-1); } private void moveSelection(int delta) { moveSelection(getNode().getSelectionModel().getSelectedIndex(), delta); } private void moveSelection(int startIndex, int delta) { final TabPane tabPane = getNode(); int tabIndex = findValidTab(startIndex, delta); if (tabIndex > -1) { final SelectionModel selectionModel = tabPane.getSelectionModel(); selectionModel.select(tabIndex); } tabPane.requestFocus(); } private int findValidTab(int startIndex, int delta) { final TabPane tabPane = getNode(); final List tabs = tabPane.getTabs(); final int max = tabs.size(); int index = startIndex; do { index = nextIndex(index + delta, max); Tab tab = tabs.get(index); if (tab != null && !tab.isDisable()) { return index; } } while (index != startIndex); return -1; } private int nextIndex(int value, int max) { final int min = 0; int r = value % max; if (r > min && max < min) { r = r + max - min; } else if (r < min && max > min) { r = r + max - min; } return r; } }