8256373: [Windows/HiDPI] The Frame#setBounds does not work in a minimized state
Reviewed-by: kizune, aivanov
This commit is contained in:
parent
0eaf0bbe26
commit
b5ce8af3d7
@ -651,31 +651,38 @@ MsgRouting AwtFrame::WmNcMouseDown(WPARAM hitTest, int x, int y, int button) {
|
||||
|
||||
// Override AwtWindow::Reshape() to handle minimized/maximized
|
||||
// frames (see 6525850, 4065534)
|
||||
void AwtFrame::Reshape(int x, int y, int width, int height)
|
||||
void AwtFrame::Reshape(int x, int y, int w, int h)
|
||||
{
|
||||
if (isIconic()) {
|
||||
// normal AwtComponent::Reshape will not work for iconified windows so...
|
||||
POINT pt = {x + w / 2, y + h / 2};
|
||||
Devices::InstanceAccess devices;
|
||||
HMONITOR monitor = MonitorFromPoint(pt, MONITOR_DEFAULTTONEAREST);
|
||||
int screen = AwtWin32GraphicsDevice::GetScreenFromHMONITOR(monitor);
|
||||
AwtWin32GraphicsDevice *device = devices->GetDevice(screen);
|
||||
// Try to set the correct size and jump to the correct location, even if
|
||||
// it is on the different monitor. Note that for the "size" we use the
|
||||
// current monitor, so the WM_DPICHANGED will adjust it for the "target"
|
||||
// monitor.
|
||||
MONITORINFO *miInfo = AwtWin32GraphicsDevice::GetMonitorInfo(screen);
|
||||
x = device == NULL ? x : device->ScaleUpAbsX(x);
|
||||
y = device == NULL ? y : device->ScaleUpAbsY(y);
|
||||
w = ScaleUpX(w);
|
||||
h = ScaleUpY(h);
|
||||
// SetWindowPlacement takes workspace coordinates, but if taskbar is at
|
||||
// top/left of screen, workspace coords != screen coords, so offset by
|
||||
// workspace origin
|
||||
x = x - (miInfo->rcWork.left - miInfo->rcMonitor.left);
|
||||
y = y - (miInfo->rcWork.top - miInfo->rcMonitor.top);
|
||||
WINDOWPLACEMENT wp;
|
||||
POINT ptMinPosition = {x,y};
|
||||
POINT ptMaxPosition = {0,0};
|
||||
RECT rcNormalPosition = {x,y,x+width,y+height};
|
||||
RECT rcWorkspace;
|
||||
HWND hWndDesktop = GetDesktopWindow();
|
||||
HWND hWndSelf = GetHWnd();
|
||||
|
||||
// SetWindowPlacement takes workspace coordinates, but
|
||||
// if taskbar is at top of screen, workspace coords !=
|
||||
// screen coords, so offset by workspace origin
|
||||
VERIFY(::SystemParametersInfo(SPI_GETWORKAREA, 0, (PVOID)&rcWorkspace, 0));
|
||||
::OffsetRect(&rcNormalPosition, -rcWorkspace.left, -rcWorkspace.top);
|
||||
|
||||
::ZeroMemory(&wp, sizeof(WINDOWPLACEMENT));
|
||||
// set the window size for when it is not-iconified
|
||||
wp.length = sizeof(wp);
|
||||
wp.flags = WPF_SETMINPOSITION;
|
||||
wp.showCmd = IsVisible() ? SW_SHOWMINIMIZED : SW_HIDE;
|
||||
wp.ptMinPosition = ptMinPosition;
|
||||
wp.ptMaxPosition = ptMaxPosition;
|
||||
wp.rcNormalPosition = rcNormalPosition;
|
||||
wp.ptMinPosition = {x, y};
|
||||
wp.ptMaxPosition = {0, 0};
|
||||
wp.rcNormalPosition = {x, y, x + w, y + h};
|
||||
|
||||
// If the call is not guarded with ignoreWmSize,
|
||||
// a regression for bug 4851435 appears.
|
||||
@ -683,7 +690,7 @@ void AwtFrame::Reshape(int x, int y, int width, int height)
|
||||
// changing the iconified state of the frame
|
||||
// while calling the Frame.setBounds() method.
|
||||
m_ignoreWmSize = TRUE;
|
||||
::SetWindowPlacement(hWndSelf, &wp);
|
||||
::SetWindowPlacement(GetHWnd(), &wp);
|
||||
m_ignoreWmSize = FALSE;
|
||||
|
||||
return;
|
||||
@ -703,7 +710,7 @@ void AwtFrame::Reshape(int x, int y, int width, int height)
|
||||
}
|
||||
}
|
||||
|
||||
AwtWindow::Reshape(x, y, width, height);
|
||||
AwtWindow::Reshape(x, y, w, h);
|
||||
}
|
||||
|
||||
|
||||
|
@ -0,0 +1,91 @@
|
||||
/*
|
||||
* Copyright (c) 2020, 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.
|
||||
*/
|
||||
|
||||
import java.awt.Frame;
|
||||
import java.awt.GraphicsDevice;
|
||||
import java.awt.GraphicsEnvironment;
|
||||
import java.awt.Rectangle;
|
||||
import java.awt.Toolkit;
|
||||
|
||||
/**
|
||||
* @test
|
||||
* @bug 8256373
|
||||
* @key headful
|
||||
* @summary setBounds() should work if the frame is minimized
|
||||
*/
|
||||
public final class RestoreToOppositeScreen {
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
Toolkit toolkit = Toolkit.getDefaultToolkit();
|
||||
if (!toolkit.isFrameStateSupported(Frame.ICONIFIED)) {
|
||||
return;
|
||||
}
|
||||
|
||||
var ge = GraphicsEnvironment.getLocalGraphicsEnvironment();
|
||||
GraphicsDevice[] gds = ge.getScreenDevices();
|
||||
for (GraphicsDevice gd1 : gds) {
|
||||
Rectangle screen1 = gd1.getDefaultConfiguration().getBounds();
|
||||
int x1 = (int) screen1.getCenterX();
|
||||
int y1 = (int) screen1.getCenterY();
|
||||
for (GraphicsDevice gd2 : gds) {
|
||||
Rectangle screen2 = gd2.getDefaultConfiguration().getBounds();
|
||||
// tweak the (x2, y2) point so even if the screen1 and screen2
|
||||
// are the same, we will use different bounds, otherwise
|
||||
// setBounds() will be ignored
|
||||
int x2 = (int) screen2.getCenterX() - 50;
|
||||
int y2 = (int) screen2.getCenterY() - 50;
|
||||
Frame frame = new Frame();
|
||||
try {
|
||||
// show the frame on one monitor, and then move it to
|
||||
// another while the frame minimized
|
||||
frame.setBounds(x1, y1, 400, 400);
|
||||
frame.setVisible(true);
|
||||
Thread.sleep(2000);
|
||||
frame.setExtendedState(Frame.ICONIFIED);
|
||||
Thread.sleep(2000);
|
||||
Rectangle before = new Rectangle(x2, y2, 380, 380);
|
||||
frame.setBounds(before);
|
||||
Thread.sleep(2000);
|
||||
frame.setExtendedState(Frame.NORMAL);
|
||||
Thread.sleep(2000);
|
||||
Rectangle after = frame.getBounds();
|
||||
checkSize(after.x, before.x, "x");
|
||||
checkSize(after.y, before.y, "y");
|
||||
checkSize(after.width, before.width, "width");
|
||||
checkSize(after.height, before.height, "height");
|
||||
} finally {
|
||||
frame.dispose();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static void checkSize(int actual, int expected, String prop) {
|
||||
if (Math.abs(actual - expected) > 10) { // let's allow size variation,
|
||||
// the bug is reproduced anyway
|
||||
System.err.println("Expected: " + expected);
|
||||
System.err.println("Actual: " + actual);
|
||||
throw new RuntimeException(prop + " is wrong");
|
||||
}
|
||||
}
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user