# HG changeset patch # User pbansal # Date 1509358555 -19800 # Node ID fc3ec7e40a12632abf293a5e602c769f50bc685f # Parent b57efb5771d3c3a50056e6de3abf5f276b30af77 8159062: [hidpi] DnD on Windows while scaling is non-integer Reviewed-by: serb, pkbalakr Contributed-by: pankaj.b.bansal@oracle.com diff -r b57efb5771d3 -r fc3ec7e40a12 src/java.desktop/windows/native/libawt/windows/awt_Win32GraphicsDevice.cpp --- a/src/java.desktop/windows/native/libawt/windows/awt_Win32GraphicsDevice.cpp Mon Oct 30 12:16:56 2017 +0530 +++ b/src/java.desktop/windows/native/libawt/windows/awt_Win32GraphicsDevice.cpp Mon Oct 30 15:45:55 2017 +0530 @@ -632,26 +632,27 @@ int AwtWin32GraphicsDevice::ScaleUpX(int x) { - return CheckIntLimits(x * scaleX); + return ClipRound(x * scaleX); } int AwtWin32GraphicsDevice::ScaleUpY(int y) { - return CheckIntLimits(y * scaleY); + return ClipRound(y * scaleY); } int AwtWin32GraphicsDevice::ScaleDownX(int x) { - return CheckIntLimits(x / scaleX); + return ClipRound(x / scaleX); } int AwtWin32GraphicsDevice::ScaleDownY(int y) { - return CheckIntLimits(y / scaleY); + return ClipRound(y / scaleY); } -int AwtWin32GraphicsDevice::CheckIntLimits(double value) +int AwtWin32GraphicsDevice::ClipRound(double value) { + value -= 0.5; if (value < INT_MIN) { return INT_MIN; diff -r b57efb5771d3 -r fc3ec7e40a12 src/java.desktop/windows/native/libawt/windows/awt_Win32GraphicsDevice.h --- a/src/java.desktop/windows/native/libawt/windows/awt_Win32GraphicsDevice.h Mon Oct 30 12:16:56 2017 +0530 +++ b/src/java.desktop/windows/native/libawt/windows/awt_Win32GraphicsDevice.h Mon Oct 30 15:45:55 2017 +0530 @@ -119,7 +119,7 @@ float scaleY; static HDC MakeDCFromMonitor(HMONITOR); - static int CheckIntLimits(double value); + static int ClipRound(double value); }; #endif AWT_WIN32GRAPHICSDEVICE_H diff -r b57efb5771d3 -r fc3ec7e40a12 test/jdk/java/awt/dnd/DnDTestWithHIDPI/DragTestWithHIDPI.java --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/test/jdk/java/awt/dnd/DnDTestWithHIDPI/DragTestWithHIDPI.java Mon Oct 30 15:45:55 2017 +0530 @@ -0,0 +1,209 @@ +/* + * 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 8159062 + * @summary Tests DnD property with HIDPI scale set to non-interger value 2.5 + * @run main/othervm -Dsun.java2d.uiScale=2.5 DragTestWithHIDPI + */ + +import javax.swing.JList; +import javax.swing.TransferHandler; +import javax.swing.JFrame; +import javax.swing.JComponent; +import javax.swing.SwingUtilities; + +import java.awt.Robot; +import java.awt.BorderLayout; +import java.awt.Point; +import java.awt.Dimension; +import java.awt.MouseInfo; +import java.awt.event.InputEvent; + +public class DragTestWithHIDPI extends TransferHandler { + + private static boolean didDrag = false; + private static int threshold = 10; + private static int DEFAULT_DELAY = 550; + + private Robot robot = null; + private static JList list = null; + private static Point listLocation = null; + private static Dimension listSize = null; + private static JFrame f = null; + + private enum Direction { + RIGHT, LEFT, BOTTOM, TOP + } + + public static void main(String[] args) throws Exception { + DragTestWithHIDPI test = new DragTestWithHIDPI(); + + // set the mouse move drag threshold + System.setProperty("awt.dnd.drag.threshold", String.valueOf(threshold)); + + test.createGUI(); + test.doTest(); + System.out.println("Test Passed"); + test.disposeGUI(); + } + + public void exportAsDrag(JComponent comp, InputEvent e, int action) { + didDrag = true; + } + + public DragTestWithHIDPI() { + super("foreground"); + } + + private void createGUI() throws Exception{ + SwingUtilities.invokeAndWait(() -> { + String[] listData = + new String[]{"Pacific Ocean", "Atlantic Ocean", "Indian Ocean", + "Arctic Ocean"}; + list = new JList(listData); + list.setDragEnabled(true); + list.setTransferHandler(new DragTestWithHIDPI()); + + f = new JFrame("DragTestWithHIDPI"); + f.getContentPane().add(list, BorderLayout.CENTER); + f.pack(); + f.toFront(); + f.setVisible(true); + + listLocation = list.getLocationOnScreen(); + listSize = list.getSize(); + }); + } + + private void disposeGUI () throws Exception { + SwingUtilities.invokeAndWait(() -> { + f.dispose(); + }); + } + + private void doTest() throws Exception { + + robot = new Robot(); + robot.waitForIdle(); + + for (Direction direction : Direction.values()) { + //Drag should not start only by moving (threshold - 1) pixels + didDrag = false; + test(threshold - 1, direction); + if (didDrag) { + disposeGUI(); + throw new RuntimeException( + "Shouldn't start drag until > " + threshold + + " pixels " + " while moving " + direction); + } + + // Drag should not start only by moving threshold pixels + didDrag = false; + test(threshold, direction); + if (didDrag) { + disposeGUI(); + throw new RuntimeException( + "Shouldn't start drag until > " + threshold + + " pixels" + " while moving " + direction); + } + + // Drag should start after moving threshold + 1 pixel + didDrag = false; + test(threshold + 1, direction); + if (!didDrag) { + disposeGUI(); + throw new RuntimeException( + "Should start drag after > " + threshold + " pixels" + + " while moving " + direction); + } + } + } + + private void test(int threshold, Direction direction) { + clickMouseOnList(InputEvent.BUTTON1_DOWN_MASK); + Point p = MouseInfo.getPointerInfo().getLocation(); + robot.mousePress(InputEvent.BUTTON1_DOWN_MASK); + robot.delay(DEFAULT_DELAY); + glide(p, direction, threshold); + robot.mouseRelease(InputEvent.BUTTON1_DOWN_MASK); + } + + private void glide(Point p, Direction direction, int toAdd) { + switch (direction) { + case RIGHT: + // move towards right + glide(p.x, p.y, p.x + toAdd, p.y); + break; + case LEFT: + // move towards left + glide(p.x, p.y, p.x - toAdd, p.y); + break; + case BOTTOM: + // move towards bottom + glide(p.x, p.y, p.x, p.y + toAdd); + break; + case TOP: + // move towards top + glide(p.x, p.y, p.x, p.y - toAdd); + break; + + } + } + + /* + Some utilities functions from JRobot class. + */ + private void moveMouseToList() { + int x = listLocation.x + listSize.width/2; + int y = listLocation.y + listSize.height/2; + robot.mouseMove(x, y); + robot.delay(DEFAULT_DELAY); + } + + private void clickMouse(int buttons) { + robot.mousePress(buttons); + robot.mouseRelease(buttons); + robot.delay(DEFAULT_DELAY); + } + + private void clickMouseOnList(int buttons) { + moveMouseToList(); + clickMouse(buttons); + } + + private void glide(int x0, int y0, int x1, int y1) { + float dmax = (float)Math.max(Math.abs(x1 - x0), Math.abs(y1 - y0)); + float dx = (x1 - x0) / dmax; + float dy = (y1 - y0) / dmax; + + robot.mouseMove(x0, y0); + for (int i=1; i<=dmax; i++) { + robot.mouseMove((int)(x0 + dx*i), (int)(y0 + dy*i)); + } + robot.delay(DEFAULT_DELAY); + } +} +