/* * Copyright (c) 1997, 2014, 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.Font; import java.awt.Graphics2D; import java.awt.GraphicsConfiguration; import java.awt.GraphicsDevice; import java.awt.GraphicsEnvironment; import java.awt.Insets; import java.awt.Rectangle; import java.awt.Toolkit; import java.awt.font.TextAttribute; import java.awt.image.BufferedImage; import java.awt.peer.ComponentPeer; import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.FilenameFilter; import java.io.InputStreamReader; import java.io.IOException; import java.text.AttributedCharacterIterator; import java.util.ArrayList; import java.util.HashSet; import java.util.Iterator; import java.util.Locale; import java.util.Map; import java.util.NoSuchElementException; import java.util.Set; import java.util.StringTokenizer; import java.util.TreeMap; import java.util.Vector; import java.util.concurrent.ConcurrentHashMap; import sun.awt.AppContext; import sun.awt.DisplayChangedListener; import sun.awt.FontConfiguration; import sun.awt.SunDisplayChanger; import sun.font.CompositeFontDescriptor; import sun.font.Font2D; import sun.font.FontManager; import sun.font.FontManagerFactory; import sun.font.FontManagerForSGE; import sun.font.NativeFont; import java.security.AccessController; import sun.security.action.GetPropertyAction; /** * 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 { public static boolean isOpenSolaris; private static Font defaultFont; private static final boolean uiScaleEnabled; private static final double debugScale; static { uiScaleEnabled = "true".equals(AccessController.doPrivileged( new GetPropertyAction("sun.java2d.uiScale.enabled", "true"))); debugScale = uiScaleEnabled ? getScaleFactor("sun.java2d.uiScale") : -1; } public SunGraphicsEnvironment() { java.security.AccessController.doPrivileged( new java.security.PrivilegedAction<Object>() { public Object run() { String osName = System.getProperty("os.name"); if ("SunOS".equals(osName)) { String version = System.getProperty("os.version", "0.0"); try { float ver = Float.parseFloat(version); if (ver > 5.10f) { File f = new File("/etc/release"); FileInputStream fis = new FileInputStream(f); InputStreamReader isr = new InputStreamReader(fis, "ISO-8859-1"); BufferedReader br = new BufferedReader(isr); String line = br.readLine(); if (line.indexOf("OpenSolaris") >= 0) { isOpenSolaris = true; } else { /* We are using isOpenSolaris as meaning * we know the Solaris commercial fonts aren't * present. "Solaris Next" (03/10) did not * include these even though its was not * OpenSolaris. Need to revisit how this is * handled but for now as in 6ux, we'll use * the test for a standard font resource as * being an indicator as to whether we need * to treat this as OpenSolaris from a font * config perspective. */ String courierNew = "/usr/openwin/lib/X11/fonts/TrueType/CourierNew.ttf"; File courierFile = new File(courierNew); isOpenSolaris = !courierFile.exists(); } fis.close(); } } catch (Exception e) { } } /* Establish the default font to be used by SG2D etc */ defaultFont = new Font(Font.DIALOG, Font.PLAIN, 12); return null; } }); } protected GraphicsDevice[] screens; /** * 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<String, String> map = fm.getCreatedFontFamilyNames(); if (map == null || map.size() == 0) { return installed; } else { for (int i=0; i<installed.length; i++) { map.put(installed[i].toLowerCase(requestedLocale), installed[i]); } String[] retval = new String[map.size()]; Object [] keyNames = map.keySet().toArray(); for (int i=0; i < keyNames.length; i++) { retval[i] = map.get(keyNames[i]); } return retval; } } public String[] getAvailableFontFamilyNames() { return getAvailableFontFamilyNames(Locale.getDefault()); } /** * Return the bounds of a GraphicsDevice, less its screen insets. * See also java.awt.GraphicsEnvironment.getUsableBounds(); */ public static Rectangle getUsableBounds(GraphicsDevice gd) { GraphicsConfiguration gc = gd.getDefaultConfiguration(); Insets insets = Toolkit.getDefaultToolkit().getScreenInsets(gc); Rectangle usableBounds = gc.getBounds(); usableBounds.x += insets.left; usableBounds.y += insets.top; usableBounds.width -= (insets.left + insets.right); usableBounds.height -= (insets.top + insets.bottom); return usableBounds; } /** * From the DisplayChangedListener interface; called * when the display mode has been changed. */ public void displayChanged() { // notify screens in device array to do display update stuff for (GraphicsDevice gd : getScreenDevices()) { if (gd instanceof DisplayChangedListener) { ((DisplayChangedListener) gd).displayChanged(); } } // notify SunDisplayChanger list (e.g. VolatileSurfaceManagers and // SurfaceDataProxies) about the display change event displayChanger.notifyListeners(); } /** * Part of the DisplayChangedListener interface: * propagate this event to listeners */ public void paletteChanged() { displayChanger.notifyPaletteChanged(); } /** * Returns true when the display is local, false for remote displays. * * @return true when the display is local, false for remote displays */ public abstract boolean isDisplayLocal(); /* * ----DISPLAY CHANGE SUPPORT---- */ protected SunDisplayChanger displayChanger = new SunDisplayChanger(); /** * Add a DisplayChangeListener to be notified when the display settings * are changed. */ public void addDisplayChangedListener(DisplayChangedListener client) { displayChanger.add(client); } /** * Remove a DisplayChangeListener from Win32GraphicsEnvironment */ public void removeDisplayChangedListener(DisplayChangedListener client) { displayChanger.remove(client); } /* * ----END DISPLAY CHANGE SUPPORT---- */ /** * Returns true if FlipBufferStrategy with COPIED buffer contents * is preferred for this peer's GraphicsConfiguration over * BlitBufferStrategy, false otherwise. * * The reason FlipBS could be preferred is that in some configurations * an accelerated copy to the screen is supported (like Direct3D 9) * * @return true if flip strategy should be used, false otherwise */ public boolean isFlipStrategyPreferred(ComponentPeer peer) { return false; } public static boolean isUIScaleEnabled() { return uiScaleEnabled; } public static double getDebugScale() { return debugScale; } public static double getScaleFactor(String propertyName) { String scaleFactor = AccessController.doPrivileged( new GetPropertyAction(propertyName, "-1")); if (scaleFactor == null || scaleFactor.equals("-1")) { return -1; } try { double units = 1.0; if (scaleFactor.endsWith("x")) { scaleFactor = scaleFactor.substring(0, scaleFactor.length() - 1); } else if (scaleFactor.endsWith("dpi")) { units = 96; scaleFactor = scaleFactor.substring(0, scaleFactor.length() - 3); } else if (scaleFactor.endsWith("%")) { units = 100; scaleFactor = scaleFactor.substring(0, scaleFactor.length() - 1); } double scale = Double.parseDouble(scaleFactor); return scale <= 0 ? -1 : scale / units; } catch (NumberFormatException ignored) { return -1; } } }