1 /*
   2  * Copyright (c) 2011, 2014, Oracle and/or its affiliates. All rights reserved.
   3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
   4  *
   5  * This code is free software; you can redistribute it and/or modify it
   6  * under the terms of the GNU General Public License version 2 only, as
   7  * published by the Free Software Foundation.  Oracle designates this
   8  * particular file as subject to the "Classpath" exception as provided
   9  * by Oracle in the LICENSE file that accompanied this code.
  10  *
  11  * This code is distributed in the hope that it will be useful, but WITHOUT
  12  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
  13  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  14  * version 2 for more details (a copy is included in the LICENSE file that
  15  * accompanied this code).
  16  *
  17  * You should have received a copy of the GNU General Public License version
  18  * 2 along with this work; if not, write to the Free Software Foundation,
  19  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
  20  *
  21  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
  22  * or visit www.oracle.com if you need additional information or have any
  23  * questions.
  24  */
  25 
  26 package sun.font;
  27 
  28 import java.awt.*;
  29 import java.io.File;
  30 import java.security.AccessController;
  31 import java.security.PrivilegedAction;
  32 import java.util.ArrayList;
  33 import java.util.HashMap;
  34 import java.util.Hashtable;
  35 import java.util.Locale;
  36 import java.util.TreeMap;
  37 import java.util.Vector;
  38 
  39 import javax.swing.plaf.FontUIResource;
  40 
  41 import sun.awt.FontConfiguration;
  42 import sun.awt.HeadlessToolkit;
  43 import sun.awt.util.ThreadGroupUtils;
  44 import sun.lwawt.macosx.*;
  45 
  46 public class CFontManager extends SunFontManager {
  47     private FontConfigManager fcManager = null;
  48     private static Hashtable<String, Font2D> genericFonts = new Hashtable<String, Font2D>();
  49 
  50     @Override
  51     protected FontConfiguration createFontConfiguration() {
  52         FontConfiguration fc = new CFontConfiguration(this);
  53         fc.init();
  54         return fc;
  55     }
  56 
  57     @Override
  58     public FontConfiguration createFontConfiguration(boolean preferLocaleFonts,
  59                                                      boolean preferPropFonts)
  60     {
  61         return new CFontConfiguration(this, preferLocaleFonts, preferPropFonts);
  62     }
  63 
  64     private static String[] defaultPlatformFont = null;
  65 
  66     /*
  67      * Returns an array of two strings. The first element is the
  68      * name of the font. The second element is the file name.
  69      */
  70     @Override
  71     public synchronized String[] getDefaultPlatformFont() {
  72         if (defaultPlatformFont == null) {
  73             defaultPlatformFont = new String[2];
  74             defaultPlatformFont[0] = "Lucida Grande";
  75             defaultPlatformFont[1] = "/System/Library/Fonts/LucidaGrande.ttc";
  76         }
  77         return defaultPlatformFont;
  78     }
  79 
  80     // This is a way to register any kind of Font2D, not just files and composites.
  81     public static Font2D[] getGenericFonts() {
  82         return genericFonts.values().toArray(new Font2D[0]);
  83     }
  84 
  85     public Font2D registerGenericFont(Font2D f)
  86     {
  87         return registerGenericFont(f, false);
  88     }
  89     public Font2D registerGenericFont(Font2D f, boolean logicalFont)
  90     {
  91         int rank = 4;
  92 
  93         String fontName = f.fullName;
  94         String familyName = f.familyName;
  95 
  96         if (fontName == null || "".equals(fontName)) {
  97             return null;
  98         }
  99 
 100         // logical fonts always need to be added to the family
 101         // plus they never need to be added to the generic font list
 102         // or the fullNameToFont table since they are covers for
 103         // already existing fonts in this list
 104         if (logicalFont || !genericFonts.containsKey(fontName)) {
 105             if (FontUtilities.debugFonts()) {
 106                 FontUtilities.getLogger().info("Add to Family "+familyName +
 107                     ", Font " + fontName + " rank="+rank);
 108             }
 109             FontFamily family = FontFamily.getFamily(familyName);
 110             if (family == null) {
 111                 family = new FontFamily(familyName, false, rank);
 112                 family.setFont(f, f.style);
 113             } else if (family.getRank() >= rank) {
 114                 family.setFont(f, f.style);
 115             }
 116             if (!logicalFont)
 117             {
 118                 genericFonts.put(fontName, f);
 119                 fullNameToFont.put(fontName.toLowerCase(Locale.ENGLISH), f);
 120             }
 121             return f;
 122         } else {
 123             return genericFonts.get(fontName);
 124         }
 125     }
 126 
 127     @Override
 128     public Font2D[] getRegisteredFonts() {
 129         Font2D[] regFonts = super.getRegisteredFonts();
 130 
 131         // Add in the Mac OS X native fonts
 132         Font2D[] genericFonts = getGenericFonts();
 133         Font2D[] allFonts = new Font2D[regFonts.length+genericFonts.length];
 134         System.arraycopy(regFonts, 0, allFonts, 0, regFonts.length);
 135         System.arraycopy(genericFonts, 0, allFonts, regFonts.length, genericFonts.length);
 136 
 137         return allFonts;
 138     }
 139 
 140     @Override
 141     protected void addNativeFontFamilyNames(TreeMap<String, String> familyNames, Locale requestedLocale) {
 142         Font2D[] genericfonts = getGenericFonts();
 143         for (int i=0; i < genericfonts.length; i++) {
 144             if (!(genericfonts[i] instanceof NativeFont)) {
 145                 String name = genericfonts[i].getFamilyName(requestedLocale);
 146                 familyNames.put(name.toLowerCase(requestedLocale), name);
 147             }
 148         }
 149     }
 150 
 151     @Override
 152     public Font2D createFont2D(File fontFile, int fontFormat, boolean isCopy, CreatedFontTracker tracker) throws FontFormatException {
 153 
 154     String fontFilePath = fontFile.getPath();
 155     Font2D font2D = null;
 156     final File fFile = fontFile;
 157     final CreatedFontTracker _tracker = tracker;
 158     try {
 159         switch (fontFormat) {
 160             case Font.TRUETYPE_FONT:
 161                         font2D = new TrueTypeFont(fontFilePath, null, 0, true);
 162                 break;
 163             case Font.TYPE1_FONT:
 164                         font2D = new Type1Font(fontFilePath, null, isCopy);
 165                 break;
 166             default:
 167                 throw new FontFormatException("Unrecognised Font Format");
 168         }
 169     } catch (FontFormatException e) {
 170         if (isCopy) {
 171             java.security.AccessController.doPrivileged(
 172                     new java.security.PrivilegedAction<Object>() {
 173                         public Object run() {
 174                             if (_tracker != null) {
 175                                 _tracker.subBytes((int)fFile.length());
 176                             }
 177                             fFile.delete();
 178                             return null;
 179                         }
 180                     });
 181         }
 182         throw(e);
 183     }
 184     if (isCopy) {
 185         FileFont.setFileToRemove(font2D, fontFile, tracker);
 186         synchronized (FontManager.class) {
 187 
 188             if (tmpFontFiles == null) {
 189                 tmpFontFiles = new Vector<File>();
 190             }
 191             tmpFontFiles.add(fontFile);
 192 
 193             if (fileCloser == null) {
 194                 final Runnable fileCloserRunnable = new Runnable() {
 195                     public void run() {
 196                         java.security.AccessController.doPrivileged(
 197                                 new java.security.PrivilegedAction<Object>() {
 198                                     public Object run() {
 199 
 200                                         for (int i=0;i<CHANNELPOOLSIZE;i++) {
 201                                             if (fontFileCache[i] != null) {
 202                                                 try {
 203                                                     fontFileCache[i].close();
 204                                                 } catch (Exception e) {}
 205                                             }
 206                                         }
 207                                         if (tmpFontFiles != null) {
 208                                             File[] files = new File[tmpFontFiles.size()];
 209                                             files = tmpFontFiles.toArray(files);
 210                                             for (int f=0; f<files.length;f++) {
 211                                                 try {
 212                                                     files[f].delete();
 213                                                 } catch (Exception e) {}
 214                                             }
 215                                         }
 216                                         return null;
 217                                     }
 218                                 });
 219                     }
 220                 };
 221                 AccessController.doPrivileged(
 222                         (PrivilegedAction<Void>) () -> {
 223                             /* The thread must be a member of a thread group
 224                              * which will not get GCed before VM exit.
 225                              * Make its parent the top-level thread group.
 226                              */
 227                             ThreadGroup rootTG = ThreadGroupUtils.getRootThreadGroup();
 228                             fileCloser = new Thread(rootTG, fileCloserRunnable);
 229                             fileCloser.setContextClassLoader(null);
 230                             Runtime.getRuntime().addShutdownHook(fileCloser);
 231                             return null;
 232                         }
 233                 );
 234                 }
 235             }
 236         }
 237         return font2D;
 238     }
 239 
 240     /*
 241     public synchronized FontConfigManager getFontConfigManager() {
 242         if (fcManager  == null) {
 243             fcManager = new FontConfigManager();
 244         }
 245         return fcManager;
 246     }
 247     */
 248 
 249     protected void registerFontsInDir(String dirName, boolean useJavaRasterizer, int fontRank, boolean defer, boolean resolveSymLinks) {
 250         loadNativeDirFonts(dirName);
 251         super.registerFontsInDir(dirName, useJavaRasterizer, fontRank, defer, resolveSymLinks);
 252     }
 253 
 254     private native void loadNativeDirFonts(String dirName);
 255     private native void loadNativeFonts();
 256 
 257     void registerFont(String fontName, String fontFamilyName) {
 258         final CFont font = new CFont(fontName, fontFamilyName);
 259 
 260         registerGenericFont(font);
 261 
 262         if ((font.getStyle() & Font.ITALIC) == 0) {
 263             registerGenericFont(font.createItalicVariant(), true);
 264         }
 265     }
 266 
 267     Object waitForFontsToBeLoaded  = new Object();
 268     public void loadFonts()
 269     {
 270         synchronized(waitForFontsToBeLoaded)
 271         {
 272             super.loadFonts();
 273             java.security.AccessController.doPrivileged(
 274                 new java.security.PrivilegedAction<Object>() {
 275                     public Object run() {
 276                         loadNativeFonts();
 277                         return null;
 278                     }
 279                 }
 280             );
 281 
 282             String defaultFont = "Lucida Grande";
 283             String defaultFallback = "Lucida Sans";
 284 
 285             setupLogicalFonts("Dialog", defaultFont, defaultFallback);
 286             setupLogicalFonts("Serif", "Times", "Lucida Bright");
 287             setupLogicalFonts("SansSerif", defaultFont, defaultFallback);
 288             setupLogicalFonts("Monospaced", "Menlo", "Lucida Sans Typewriter");
 289             setupLogicalFonts("DialogInput", defaultFont, defaultFallback);
 290         }
 291     }
 292 
 293     protected void setupLogicalFonts(String logicalName, String realName, String fallbackName) {
 294         FontFamily realFamily = getFontFamilyWithExtraTry(logicalName, realName, fallbackName);
 295 
 296         cloneStyledFont(realFamily, logicalName, Font.PLAIN);
 297         cloneStyledFont(realFamily, logicalName, Font.BOLD);
 298         cloneStyledFont(realFamily, logicalName, Font.ITALIC);
 299         cloneStyledFont(realFamily, logicalName, Font.BOLD | Font.ITALIC);
 300     }
 301 
 302     protected FontFamily getFontFamilyWithExtraTry(String logicalName, String realName, String fallbackName){
 303         FontFamily family = getFontFamily(realName, fallbackName);
 304         if (family != null) return family;
 305 
 306         // at this point, we recognize that we probably needed a fallback font
 307         super.loadFonts();
 308 
 309         family = getFontFamily(realName, fallbackName);
 310         if (family != null) return family;
 311 
 312         System.err.println("Warning: the fonts \"" + realName + "\" and \"" + fallbackName + "\" are not available for the Java logical font \"" + logicalName + "\", which may have unexpected appearance or behavior. Re-enable the \""+ realName +"\" font to remove this warning.");
 313         return null;
 314     }
 315 
 316     protected FontFamily getFontFamily(String realName, String fallbackName){
 317         FontFamily family = FontFamily.getFamily(realName);
 318         if (family != null) return family;
 319 
 320         family = FontFamily.getFamily(fallbackName);
 321         if (family != null){
 322             System.err.println("Warning: the font \"" + realName + "\" is not available, so \"" + fallbackName + "\" has been substituted, but may have unexpected appearance or behavor. Re-enable the \""+ realName +"\" font to remove this warning.");
 323             return family;
 324         }
 325 
 326         return null;
 327     }
 328 
 329     protected boolean cloneStyledFont(FontFamily realFamily, String logicalFamilyName, int style) {
 330         if (realFamily == null) return false;
 331 
 332         Font2D realFont = realFamily.getFontWithExactStyleMatch(style);
 333         if (realFont == null || !(realFont instanceof CFont)) return false;
 334 
 335         CFont newFont = new CFont((CFont)realFont, logicalFamilyName);
 336         registerGenericFont(newFont, true);
 337 
 338         return true;
 339     }
 340 
 341     @Override
 342     public String getFontPath(boolean noType1Fonts) {
 343         // In the case of the Cocoa toolkit, since we go through NSFont, we don't need to register /Library/Fonts
 344         Toolkit tk = Toolkit.getDefaultToolkit();
 345         if (tk instanceof HeadlessToolkit) {
 346             tk = ((HeadlessToolkit)tk).getUnderlyingToolkit();
 347         }
 348         if (tk instanceof LWCToolkit) {
 349             return "";
 350         }
 351 
 352         // X11 case
 353         return "/Library/Fonts";
 354     }
 355 
 356     @Override
 357     protected FontUIResource getFontConfigFUIR(
 358             String family, int style, int size)
 359     {
 360         String mappedName = FontUtilities.mapFcName(family);
 361         if (mappedName == null) {
 362             mappedName = "sansserif";
 363         }
 364         return new FontUIResource(mappedName, style, size);
 365     }
 366 
 367     // Only implemented on Windows
 368     @Override
 369     protected void populateFontFileNameMap(HashMap<String, String> fontToFileMap, HashMap<String, String> fontToFamilyNameMap,
 370             HashMap<String, ArrayList<String>> familyToFontListMap, Locale locale) {}
 371 }