/* * Copyright (c) 1997, 2024, 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. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * 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. */ package sun.java2d; import java.awt.AWTError; import java.awt.Color; import java.awt.Dimension; import java.awt.Font; import java.awt.Graphics2D; import java.awt.GraphicsConfiguration; import java.awt.GraphicsDevice; import java.awt.GraphicsEnvironment; import java.awt.Insets; import java.awt.Point; import java.awt.Rectangle; import java.awt.Toolkit; import java.awt.geom.AffineTransform; import java.awt.image.BufferedImage; import java.awt.peer.ComponentPeer; import java.util.Locale; import java.util.TreeMap; import sun.awt.DisplayChangedListener; import sun.awt.SunDisplayChanger; import sun.font.FontManager; import sun.font.FontManagerFactory; import sun.font.FontManagerForSGE; import sun.font.FontUtilities; import sun.java2d.pipe.Region; /** * This is an implementation of a GraphicsEnvironment object for the * default local GraphicsEnvironment. * * @see GraphicsDevice * @see GraphicsConfiguration */ public abstract class SunGraphicsEnvironment extends GraphicsEnvironment implements DisplayChangedListener { /** Establish the default font to be used by SG2D. */ private final Font defaultFont = new Font(Font.DIALOG, Font.PLAIN, 12); private static final Object UI_SCALE_LOCK = new Object(); private static boolean uiScaleEnabled; private static Boolean uiScaleEnabled_overridden; private static final double debugScale; static { String uiScaleEnabledPropValue = System.getProperty("sun.java2d.uiScale.enabled", "true"); uiScaleEnabled = FontUtilities.isMacOSX || ("true".equals(uiScaleEnabledPropValue) && (isWindows_8_1_orUpper() || FontUtilities.isLinux)); if (uiScaleEnabled && FontUtilities.isWindows) { System.setProperty("swing.bufferPerWindow", "false"); // todo: until JRE-489 is fixed } debugScale = uiScaleEnabled ? getScaleFactor("sun.java2d.uiScale") : -1; } protected enum LogDisplay { ADDED, REMOVED, CHANGED; public static final boolean ENABLED = Boolean.getBoolean("sun.java2d.logDisplays"); public void log(int id, Object bounds, double scale) { if (!ENABLED) return; System.out.println("DISPLAY " + this + ": #" + id + ", " + bounds + ", scale=" + scale); } } protected GraphicsDevice[] screens; private static boolean isWindows_8_1_orUpper() { if (!FontUtilities.isWindows) return false; String osVersion = System.getProperty("os.version"); if (osVersion == null) return false; String[] parts = osVersion.split("\\."); if (parts.length < 1) return false; try { int majorVer = Integer.parseInt(parts[0]); if (majorVer > 6) return true; if (majorVer < 6) return false; if (parts.length < 2) return false; int minorVer = Integer.parseInt(parts[1]); if (minorVer >= 3) return true; } catch (NumberFormatException ignore) { } return false; } /** * Returns an array of all of the screen devices. */ public synchronized GraphicsDevice[] getScreenDevices() { GraphicsDevice[] ret = screens; if (ret == null) { int num = getNumScreens(); ret = new GraphicsDevice[num]; for (int i = 0; i < num; i++) { ret[i] = makeScreenDevice(i); } screens = ret; } return ret; } /** * Returns the number of screen devices of this graphics environment. * * @return the number of screen devices of this graphics environment */ protected abstract int getNumScreens(); /** * Create and return the screen device with the specified number. The * device with number {@code 0} will be the default device (returned * by {@link #getDefaultScreenDevice()}. * * @param screennum the number of the screen to create * * @return the created screen device */ protected abstract GraphicsDevice makeScreenDevice(int screennum); /** * Returns the default screen graphics device. */ public GraphicsDevice getDefaultScreenDevice() { GraphicsDevice[] screens = getScreenDevices(); if (screens.length == 0) { throw new AWTError("no screen devices"); } return screens[0]; } /** * Returns a Graphics2D object for rendering into the * given BufferedImage. * @throws NullPointerException if BufferedImage argument is null */ public Graphics2D createGraphics(BufferedImage img) { if (img == null) { throw new NullPointerException("BufferedImage cannot be null"); } SurfaceData sd = SurfaceData.getPrimarySurfaceData(img); return new SunGraphics2D(sd, Color.white, Color.black, defaultFont); } public static FontManagerForSGE getFontManagerForSGE() { FontManager fm = FontManagerFactory.getInstance(); return (FontManagerForSGE) fm; } /* Modifies the behaviour of a subsequent call to preferLocaleFonts() * to use Mincho instead of Gothic for dialoginput in JA locales * on windows. Not needed on other platforms. * * @deprecated as of JDK9. To be removed in a future release */ @Deprecated public static void useAlternateFontforJALocales() { getFontManagerForSGE().useAlternateFontforJALocales(); } /** * Returns all fonts available in this environment. */ public Font[] getAllFonts() { FontManagerForSGE fm = getFontManagerForSGE(); Font[] installedFonts = fm.getAllInstalledFonts(); Font[] created = fm.getCreatedFonts(); if (created == null || created.length == 0) { return installedFonts; } else { int newlen = installedFonts.length + created.length; Font [] fonts = java.util.Arrays.copyOf(installedFonts, newlen); System.arraycopy(created, 0, fonts, installedFonts.length, created.length); return fonts; } } public String[] getAvailableFontFamilyNames(Locale requestedLocale) { FontManagerForSGE fm = getFontManagerForSGE(); String[] installed = fm.getInstalledFontFamilyNames(requestedLocale); /* Use a new TreeMap as used in getInstalledFontFamilyNames * and insert all the keys in lower case, so that the sort order * is the same as the installed families. This preserves historical * behaviour and inserts new families in the right place. * It would have been marginally more efficient to directly obtain * the tree map and just insert new entries, but not so much as * to justify the extra internal interface. */ TreeMap map = fm.getCreatedFontFamilyNames(); if (map == null || map.size() == 0) { return installed; } else { for (int i=0; i