mirror of
https://github.com/openjdk/jdk.git
synced 2026-03-05 13:40:18 +00:00
8162856: JSlider thumb is twice smaller on HiDPI display
Reviewed-by: prr, serb
This commit is contained in:
parent
f270d810ad
commit
fa627e16d5
@ -657,7 +657,6 @@ class XPStyle {
|
||||
|
||||
protected void paintToImage(Component c, Image image, Graphics g,
|
||||
int w, int h, Object[] args) {
|
||||
boolean accEnabled = false;
|
||||
Skin skin = (Skin)args[0];
|
||||
Part part = skin.part;
|
||||
State state = (State)args[1];
|
||||
@ -668,6 +667,8 @@ class XPStyle {
|
||||
c = skin.component;
|
||||
}
|
||||
BufferedImage bi = (BufferedImage)image;
|
||||
w = bi.getWidth();
|
||||
h = bi.getHeight();
|
||||
|
||||
WritableRaster raster = bi.getRaster();
|
||||
DataBufferInt dbi = (DataBufferInt)raster.getDataBuffer();
|
||||
|
||||
@ -100,7 +100,9 @@ public abstract class CachedPainter {
|
||||
}
|
||||
}
|
||||
|
||||
private Image getImage(Object key, Component c, int w, int h, Object... args) {
|
||||
private Image getImage(Object key, Component c,
|
||||
int baseWidth, int baseHeight,
|
||||
int w, int h, Object... args) {
|
||||
GraphicsConfiguration config = getGraphicsConfiguration(c);
|
||||
ImageCache cache = getCache(key);
|
||||
Image image = cache.getImage(key, config, w, h, args);
|
||||
@ -127,8 +129,11 @@ public abstract class CachedPainter {
|
||||
}
|
||||
if (draw) {
|
||||
// Render to the Image
|
||||
Graphics g2 = image.getGraphics();
|
||||
paintToImage(c, image, g2, w, h, args);
|
||||
Graphics2D g2 = (Graphics2D) image.getGraphics();
|
||||
if (w != baseWidth || h != baseHeight) {
|
||||
g2.scale((double) w / baseWidth, (double) h / baseHeight);
|
||||
}
|
||||
paintToImage(c, image, g2, baseWidth, baseHeight, args);
|
||||
g2.dispose();
|
||||
}
|
||||
|
||||
@ -149,14 +154,7 @@ public abstract class CachedPainter {
|
||||
Image image = cache.getImage(key, config, w, h, args);
|
||||
|
||||
if (image == null) {
|
||||
double sx = 1;
|
||||
double sy = 1;
|
||||
if (g instanceof Graphics2D) {
|
||||
AffineTransform tx = ((Graphics2D) g).getTransform();
|
||||
sx = tx.getScaleX();
|
||||
sy = tx.getScaleY();
|
||||
}
|
||||
image = new PainterMultiResolutionCachedImage(sx, sy, w, h);
|
||||
image = new PainterMultiResolutionCachedImage(w, h);
|
||||
cache.setImage(key, config, w, h, args, image);
|
||||
}
|
||||
|
||||
@ -238,17 +236,12 @@ public abstract class CachedPainter {
|
||||
|
||||
class PainterMultiResolutionCachedImage extends AbstractMultiResolutionImage {
|
||||
|
||||
private final double scaleX;
|
||||
private final double scaleY;
|
||||
private final int baseWidth;
|
||||
private final int baseHeight;
|
||||
private Component c;
|
||||
private Object[] args;
|
||||
|
||||
public PainterMultiResolutionCachedImage(double scaleX, double scaleY,
|
||||
int baseWidth, int baseHeight) {
|
||||
this.scaleX = scaleX;
|
||||
this.scaleY = scaleY;
|
||||
public PainterMultiResolutionCachedImage(int baseWidth, int baseHeight) {
|
||||
this.baseWidth = baseWidth;
|
||||
this.baseHeight = baseHeight;
|
||||
}
|
||||
@ -272,7 +265,7 @@ public abstract class CachedPainter {
|
||||
public Image getResolutionVariant(double destWidth, double destHeight) {
|
||||
int w = (int) Math.ceil(destWidth);
|
||||
int h = (int) Math.ceil(destHeight);
|
||||
return getImage(this, c, w, h, args);
|
||||
return getImage(this, c, baseWidth, baseHeight, w, h, args);
|
||||
}
|
||||
|
||||
@Override
|
||||
@ -282,15 +275,7 @@ public abstract class CachedPainter {
|
||||
|
||||
@Override
|
||||
public java.util.List<Image> getResolutionVariants() {
|
||||
|
||||
if (scaleX == 1 && scaleY == 1) {
|
||||
return Arrays.asList(getResolutionVariant(baseWidth, baseHeight));
|
||||
}
|
||||
|
||||
return Arrays.asList(
|
||||
getResolutionVariant(baseWidth, baseHeight),
|
||||
getResolutionVariant(scaleX * baseWidth, scaleY * baseHeight)
|
||||
);
|
||||
return Arrays.asList(getResolutionVariant(baseWidth, baseHeight));
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -0,0 +1,147 @@
|
||||
/*
|
||||
* Copyright (c) 2016, 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.Dimension;
|
||||
import java.awt.Graphics2D;
|
||||
import java.awt.image.BufferedImage;
|
||||
import javax.swing.JSlider;
|
||||
import javax.swing.SwingUtilities;
|
||||
import javax.swing.UIManager;
|
||||
import javax.swing.plaf.metal.MetalLookAndFeel;
|
||||
|
||||
/*
|
||||
* @test
|
||||
* @bug 8162856
|
||||
* @summary Bad rendering of Swing UI controls with Metal L&F on HiDPI display
|
||||
* @run main MetalHiDPISliderThumbTest
|
||||
*/
|
||||
public class MetalHiDPISliderThumbTest {
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
|
||||
SwingUtilities.invokeAndWait(() -> {
|
||||
|
||||
try {
|
||||
UIManager.setLookAndFeel(new MetalLookAndFeel());
|
||||
} catch (Exception e) {
|
||||
throw new RuntimeException(e);
|
||||
}
|
||||
|
||||
if (!testSliderThumb(true)) {
|
||||
throw new RuntimeException("Horizontal Slider Thumb is not scaled!");
|
||||
}
|
||||
|
||||
if (!testSliderThumb(false)) {
|
||||
throw new RuntimeException("Vertical Slider Thumb is not scaled!");
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
private static boolean testSliderThumb(boolean horizontal) {
|
||||
int scale = 3;
|
||||
|
||||
int w = horizontal ? 100 : 20;
|
||||
int h = horizontal ? 20 : 100;
|
||||
|
||||
JSlider testSlider = new JSlider();
|
||||
testSlider.setSize(w, h);
|
||||
Dimension size = new Dimension(w, h);
|
||||
testSlider.setPreferredSize(size);
|
||||
testSlider.setMinimumSize(size);
|
||||
testSlider.setMaximumSize(size);
|
||||
testSlider.setOrientation(horizontal ? JSlider.HORIZONTAL : JSlider.VERTICAL);
|
||||
|
||||
int sw = scale * w;
|
||||
int sh = scale * h;
|
||||
|
||||
final BufferedImage img = new BufferedImage(sw, sh, BufferedImage.TYPE_INT_RGB);
|
||||
|
||||
Graphics2D g = img.createGraphics();
|
||||
g.scale(scale, scale);
|
||||
testSlider.paint(g);
|
||||
g.dispose();
|
||||
|
||||
if (horizontal) {
|
||||
int y = sh / 2;
|
||||
|
||||
int xMin = 0;
|
||||
int rgb = img.getRGB(xMin, y);
|
||||
for (int i = 0; i < sw; i++) {
|
||||
if (img.getRGB(i, y) != rgb) {
|
||||
xMin = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int xMax = sw - 1;
|
||||
rgb = img.getRGB(xMax, y);
|
||||
for (int i = sw - 1; i > 0; i--) {
|
||||
if (img.getRGB(i, y) != rgb) {
|
||||
xMax = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int d = 3 * scale;
|
||||
int xc = (xMin + xMax) / 2 - d;
|
||||
rgb = img.getRGB(xc, y);
|
||||
|
||||
for (int x = xMin + d; x < xc; x++) {
|
||||
if (img.getRGB(x, y) != rgb) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
} else {
|
||||
int x = sw / 2;
|
||||
|
||||
int yMin = 0;
|
||||
int rgb = img.getRGB(x, yMin);
|
||||
for (int i = 0; i < sh; i++) {
|
||||
if (img.getRGB(x, i) != rgb) {
|
||||
yMin = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int yMax = sh - 1;
|
||||
rgb = img.getRGB(x, yMax);
|
||||
for (int i = sh - 1; i > 0; i--) {
|
||||
if (img.getRGB(x, i) != rgb) {
|
||||
yMax = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
int d = 3 * scale;
|
||||
int yc = (yMin + yMax) / 2 - d;
|
||||
rgb = img.getRGB(x, yc);
|
||||
|
||||
for (int y = yMin + d; y < yc; y++) {
|
||||
if (img.getRGB(x, y) != rgb) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
}
|
||||
Loading…
x
Reference in New Issue
Block a user