5b80b8d7d1
Reviewed-by: serb, pkbalakr, aghaisas
258 lines
8.6 KiB
Java
258 lines
8.6 KiB
Java
/*
|
|
* Copyright (c) 2017, 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.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
/**
|
|
* @test
|
|
* @key headful
|
|
* @bug 4494085 8158366
|
|
* @summary verifies that the recognized action matches modifiers state
|
|
* @compile RecognizedActionTest.java
|
|
* @run main RecognizedActionTest
|
|
*/
|
|
|
|
import java.awt.Frame;
|
|
import java.awt.Component;
|
|
import java.awt.Robot;
|
|
import java.awt.Point;
|
|
import java.awt.Dimension;
|
|
import java.awt.AWTEvent;
|
|
import java.awt.event.AWTEventListener;
|
|
import java.awt.event.InputEvent;
|
|
import java.awt.event.KeyEvent;
|
|
import java.awt.event.MouseEvent;
|
|
import java.awt.dnd.DnDConstants;
|
|
import java.awt.dnd.DragSource;
|
|
import java.awt.dnd.DragGestureListener;
|
|
import java.awt.dnd.DragGestureEvent;
|
|
|
|
public class RecognizedActionTest implements AWTEventListener {
|
|
|
|
final Frame frame = new Frame();
|
|
boolean dragGestureRecognized = false;
|
|
int currentDragAction = DnDConstants.ACTION_NONE;
|
|
|
|
final int[] modifiers = {
|
|
0,
|
|
InputEvent.CTRL_DOWN_MASK,
|
|
InputEvent.SHIFT_DOWN_MASK,
|
|
InputEvent.SHIFT_DOWN_MASK | InputEvent.CTRL_DOWN_MASK
|
|
};
|
|
|
|
final DragSource dragSource = DragSource.getDefaultDragSource();
|
|
final DragGestureListener dragGestureListener = new DragGestureListener() {
|
|
|
|
public void dragGestureRecognized(DragGestureEvent dge) {
|
|
dragGestureRecognized = true;
|
|
|
|
if (dge.getDragAction() != currentDragAction) {
|
|
throw new RuntimeException("Expected: " +
|
|
Integer.toHexString(currentDragAction) +
|
|
" recognized: " +
|
|
Integer.toHexString(dge.getDragAction()));
|
|
}
|
|
}
|
|
};
|
|
|
|
final Object SYNC_LOCK = new Object();
|
|
final int FRAME_ACTIVATION_TIMEOUT = 2000;
|
|
final int MOUSE_RELEASE_TIMEOUT = 1000;
|
|
|
|
Component clickedComponent = null;
|
|
|
|
public void init() {
|
|
try {
|
|
frame.setTitle("Test frame");
|
|
frame.setBounds(100, 100, 200, 200);
|
|
dragSource.createDefaultDragGestureRecognizer(frame,
|
|
DnDConstants.ACTION_COPY |
|
|
DnDConstants.ACTION_MOVE |
|
|
DnDConstants.ACTION_LINK,
|
|
dragGestureListener);
|
|
|
|
frame.getToolkit().addAWTEventListener(this, AWTEvent.MOUSE_EVENT_MASK);
|
|
frame.setVisible(true);
|
|
Thread.sleep(100);
|
|
|
|
final Robot robot = new Robot();
|
|
robot.waitForIdle();
|
|
|
|
Thread.sleep(FRAME_ACTIVATION_TIMEOUT);
|
|
|
|
final Point srcPoint = frame.getLocationOnScreen();
|
|
Dimension d = frame.getSize();
|
|
srcPoint.translate(d.width / 2, d.height / 2);
|
|
|
|
if (!pointInComponent(robot, srcPoint, frame)) {
|
|
throw new RuntimeException("WARNING: Couldn't locate source frame.");
|
|
}
|
|
|
|
final Point dstPoint = new Point(srcPoint);
|
|
dstPoint.translate(d.width / 4, d.height / 4);
|
|
|
|
if (!pointInComponent(robot, dstPoint, frame)) {
|
|
throw new RuntimeException("WARNING: Couldn't locate target frame.");
|
|
}
|
|
|
|
for (int i = 0; i < modifiers.length; i++) {
|
|
currentDragAction = convertModifiersToDropAction(modifiers[i]);
|
|
dragGestureRecognized = false;
|
|
final Point curPoint = new Point(srcPoint);
|
|
robot.mouseMove(curPoint.x, curPoint.y);
|
|
|
|
switch (modifiers[i]) {
|
|
case InputEvent.SHIFT_DOWN_MASK:
|
|
robot.keyPress(KeyEvent.VK_SHIFT);
|
|
robot.waitForIdle();
|
|
break;
|
|
|
|
case InputEvent.CTRL_DOWN_MASK:
|
|
robot.keyPress(KeyEvent.VK_CONTROL);
|
|
robot.waitForIdle();
|
|
break;
|
|
|
|
case InputEvent.SHIFT_DOWN_MASK | InputEvent.CTRL_DOWN_MASK:
|
|
robot.keyPress(KeyEvent.VK_CONTROL);
|
|
robot.waitForIdle();
|
|
robot.keyPress(KeyEvent.VK_SHIFT);
|
|
robot.waitForIdle();
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
robot.mousePress(InputEvent.BUTTON1_DOWN_MASK);
|
|
Thread.sleep(100);
|
|
|
|
for (; !curPoint.equals(dstPoint) && !dragGestureRecognized;
|
|
curPoint.translate(sign(dstPoint.x - curPoint.x),
|
|
sign(dstPoint.y - curPoint.y))) {
|
|
robot.mouseMove(curPoint.x, curPoint.y);
|
|
Thread.sleep(50);
|
|
}
|
|
Thread.sleep(100);
|
|
|
|
robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK);
|
|
|
|
switch (modifiers[i]) {
|
|
case InputEvent.SHIFT_DOWN_MASK:
|
|
robot.keyRelease(KeyEvent.VK_SHIFT);
|
|
robot.waitForIdle();
|
|
break;
|
|
|
|
case InputEvent.CTRL_DOWN_MASK:
|
|
robot.keyRelease(KeyEvent.VK_CONTROL);
|
|
robot.waitForIdle();
|
|
break;
|
|
|
|
case InputEvent.SHIFT_DOWN_MASK | InputEvent.CTRL_DOWN_MASK:
|
|
robot.keyRelease(KeyEvent.VK_CONTROL);
|
|
robot.waitForIdle();
|
|
robot.keyRelease(KeyEvent.VK_SHIFT);
|
|
robot.waitForIdle();
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
Thread.sleep(100);
|
|
}
|
|
|
|
} catch (Exception e) {
|
|
e.printStackTrace();
|
|
throw new RuntimeException("The test failed.");
|
|
}
|
|
}
|
|
|
|
public int sign(int n) {
|
|
return n < 0 ? -1 : n == 0 ? 0 : 1;
|
|
}
|
|
|
|
public void reset() {
|
|
clickedComponent = null;
|
|
}
|
|
|
|
public void eventDispatched(AWTEvent e) {
|
|
if (e.getID() == MouseEvent.MOUSE_RELEASED) {
|
|
clickedComponent = (Component) e.getSource();
|
|
synchronized (SYNC_LOCK) {
|
|
SYNC_LOCK.notifyAll();
|
|
}
|
|
}
|
|
}
|
|
|
|
public boolean pointInComponent(Robot robot, Point p, Component comp)
|
|
throws InterruptedException {
|
|
reset();
|
|
robot.mouseMove(p.x, p.y);
|
|
robot.waitForIdle();
|
|
robot.mousePress(InputEvent.BUTTON1_DOWN_MASK);
|
|
robot.waitForIdle();
|
|
synchronized (SYNC_LOCK) {
|
|
robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK);
|
|
SYNC_LOCK.wait(MOUSE_RELEASE_TIMEOUT);
|
|
}
|
|
|
|
Component c = clickedComponent;
|
|
|
|
while (c != null && c != comp) {
|
|
c = c.getParent();
|
|
}
|
|
|
|
return c == comp;
|
|
}
|
|
|
|
public void dispose() {
|
|
frame.dispose();
|
|
}
|
|
|
|
public int convertModifiersToDropAction(int modifiers) {
|
|
int dropAction = DnDConstants.ACTION_NONE;
|
|
|
|
switch (modifiers & (InputEvent.SHIFT_DOWN_MASK | InputEvent.CTRL_DOWN_MASK)) {
|
|
case InputEvent.SHIFT_DOWN_MASK | InputEvent.CTRL_DOWN_MASK:
|
|
dropAction = DnDConstants.ACTION_LINK;
|
|
break;
|
|
|
|
case InputEvent.CTRL_DOWN_MASK:
|
|
dropAction = DnDConstants.ACTION_COPY;
|
|
break;
|
|
|
|
case InputEvent.SHIFT_DOWN_MASK:
|
|
dropAction = DnDConstants.ACTION_MOVE;
|
|
break;
|
|
|
|
default:
|
|
dropAction = DnDConstants.ACTION_MOVE;
|
|
break;
|
|
}
|
|
|
|
return dropAction;
|
|
}
|
|
|
|
public static void main(String args[]) {
|
|
RecognizedActionTest actionTest = new RecognizedActionTest();
|
|
actionTest.init();
|
|
actionTest.dispose();
|
|
}
|
|
}
|