/* * Copyright (c) 2010, 2014, Oracle and/or its affiliates. All rights reserved. * ORACLE PROPRIETARY/CONFIDENTIAL. Use is subject to license terms. * * * * * * * * * * * * * * * * * * * * */ package com.sun.javafx.scene.control.skin; import javafx.event.Event; import javafx.event.EventHandler; import javafx.scene.AccessibleAttribute; import javafx.scene.Node; import javafx.scene.control.ContextMenu; import javafx.scene.control.Menu; import javafx.scene.control.Skin; import javafx.scene.input.KeyCode; import javafx.scene.input.KeyEvent; import javafx.scene.layout.Region; import javafx.stage.WindowEvent; import com.sun.javafx.scene.control.behavior.TwoLevelFocusPopupBehavior; /** * Default Skin implementation for PopupMenu. Several controls use PopupMenu in * order to display items in a drop down. It deals mostly with show hide logic for * Popup based controls, and specifically in this case for PopupMenu. This is * explained in the superclass {@link PopupControlSkin}. *

* Region/css based skin implementation for PopupMenu is the {@link PopupMenuContent} * class. */ public class ContextMenuSkin implements Skin { /* need to hold a reference to popupMenu here because getSkinnable() deliberately * returns null in PopupControlSkin. */ private ContextMenu popupMenu; private final Region root; private TwoLevelFocusPopupBehavior tlFocus; // Fix for RT-18247 private final EventHandler keyListener = new EventHandler() { @Override public void handle(KeyEvent event) { if (event.getEventType() != KeyEvent.KEY_PRESSED) return; // We only care if the root container still has focus if (! root.isFocused()) return; final KeyCode code = event.getCode(); switch (code) { case ENTER: case SPACE: popupMenu.hide(); return; default: return; } } }; /***/ public ContextMenuSkin(final ContextMenu popupMenu) { this.popupMenu = popupMenu; // When a contextMenu is shown, requestFocus on its content to enable // keyboard navigation. popupMenu.addEventHandler(Menu.ON_SHOWN, new EventHandler() { @Override public void handle(Event event) { Node cmContent = popupMenu.getSkin().getNode(); if (cmContent != null) { cmContent.requestFocus(); if (cmContent instanceof ContextMenuContent) { Node accMenu = ((ContextMenuContent)cmContent).getItemsContainer(); accMenu.notifyAccessibleAttributeChanged(AccessibleAttribute.VISIBLE); } } root.addEventHandler(KeyEvent.KEY_PRESSED, keyListener); } }); popupMenu.addEventHandler(Menu.ON_HIDDEN, new EventHandler() { @Override public void handle(Event event) { Node cmContent = popupMenu.getSkin().getNode(); if (cmContent != null) cmContent.requestFocus(); root.removeEventHandler(KeyEvent.KEY_PRESSED, keyListener); } }); // For accessibility Menu.ON_HIDING does not work because isShowing is true // during the event, Menu.ON_HIDDEN does not work because the Window (in glass) // has already being disposed. The fix is to use WINDOW_HIDING (WINDOW_HIDDEN). popupMenu.addEventFilter(WindowEvent.WINDOW_HIDING, new EventHandler() { @Override public void handle(Event event) { Node cmContent = popupMenu.getSkin().getNode(); if (cmContent instanceof ContextMenuContent) { Node accMenu = ((ContextMenuContent)cmContent).getItemsContainer(); accMenu.notifyAccessibleAttributeChanged(AccessibleAttribute.VISIBLE); } } }); if (BehaviorSkinBase.IS_TOUCH_SUPPORTED && popupMenu.getStyleClass().contains("text-input-context-menu")) { root = new EmbeddedTextContextMenuContent(popupMenu); } else { root = new ContextMenuContent(popupMenu); } root.idProperty().bind(popupMenu.idProperty()); root.styleProperty().bind(popupMenu.styleProperty()); root.getStyleClass().addAll(popupMenu.getStyleClass()); // TODO needs to handle updates // Only add this if we're on an embedded platform that supports 5-button navigation if (Utils.isTwoLevelFocus()) { tlFocus = new TwoLevelFocusPopupBehavior(popupMenu); // needs to be last. } } @Override public ContextMenu getSkinnable() { return popupMenu; } @Override public Node getNode() { return root; } @Override public void dispose() { root.idProperty().unbind(); root.styleProperty().unbind(); if (tlFocus != null) tlFocus.dispose(); } }