1 /*
   2  * Copyright (c) 2011, 2019, 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 final class CFontManager extends SunFontManager {
  47     private static Hashtable<String, Font2D> genericFonts = new Hashtable<String, Font2D>();
  48 
  49     @Override
  50     protected FontConfiguration createFontConfiguration() {
  51         FontConfiguration fc = new CFontConfiguration(this);
  52         fc.init();
  53         return fc;
  54     }
  55 
  56     @Override
  57     public FontConfiguration createFontConfiguration(boolean preferLocaleFonts,
  58                                                      boolean preferPropFonts)
  59     {
  60         return new CFontConfiguration(this, preferLocaleFonts, preferPropFonts);
  61     }
  62 
  63     /*
  64      * Returns an array of two strings. The first element is the
  65      * name of the font. The second element is the file name.
  66      */
  67     @Override
  68     protected String[] getDefaultPlatformFont() {
  69         return new String[]{"Lucida Grande",
  70                             "/System/Library/Fonts/LucidaGrande.ttc"};
  71     }
  72 
  73     // This is a way to register any kind of Font2D, not just files and composites.
  74     public static Font2D[] getGenericFonts() {
  75         return genericFonts.values().toArray(new Font2D[0]);
  76     }
  77 
  78     public Font2D registerGenericFont(Font2D f)
  79     {
  80         return registerGenericFont(f, false);
  81     }
  82     public Font2D registerGenericFont(Font2D f, boolean logicalFont)
  83     {
  84         int rank = 4;
  85 
  86         String fontName = f.fullName;
  87         String familyName = f.familyName;
  88 
  89         if (fontName == null || fontName.isEmpty()) {
  90             return null;
  91         }
  92 
  93         // logical fonts always need to be added to the family
  94         // plus they never need to be added to the generic font list
  95         // or the fullNameToFont table since they are covers for
  96         // already existing fonts in this list
  97         if (logicalFont || !genericFonts.containsKey(fontName)) {
  98             if (FontUtilities.debugFonts()) {
  99                 FontUtilities.getLogger().info("Add to Family "+familyName +
 100                     ", Font " + fontName + " rank="+rank);
 101             }
 102             FontFamily family = FontFamily.getFamily(familyName);
 103             if (family == null) {
 104                 family = new FontFamily(familyName, false, rank);
 105                 family.setFont(f, f.style);
 106             } else if (family.getRank() >= rank) {
 107                 family.setFont(f, f.style);
 108             }
 109             if (!logicalFont)
 110             {
 111                 genericFonts.put(fontName, f);
 112                 fullNameToFont.put(fontName.toLowerCase(Locale.ENGLISH), f);
 113             }
 114             return f;
 115         } else {
 116             return genericFonts.get(fontName);
 117         }
 118     }
 119 
 120     @Override
 121     public Font2D[] getRegisteredFonts() {
 122         Font2D[] regFonts = super.getRegisteredFonts();
 123 
 124         // Add in the Mac OS X native fonts
 125         Font2D[] genericFonts = getGenericFonts();
 126         Font2D[] allFonts = new Font2D[regFonts.length+genericFonts.length];
 127         System.arraycopy(regFonts, 0, allFonts, 0, regFonts.length);
 128         System.arraycopy(genericFonts, 0, allFonts, regFonts.length, genericFonts.length);
 129 
 130         return allFonts;
 131     }
 132 
 133     @Override
 134     protected void addNativeFontFamilyNames(TreeMap<String, String> familyNames, Locale requestedLocale) {
 135         Font2D[] genericfonts = getGenericFonts();
 136         for (int i=0; i < genericfonts.length; i++) {
 137             if (!(genericfonts[i] instanceof NativeFont)) {
 138                 String name = genericfonts[i].getFamilyName(requestedLocale);
 139                 familyNames.put(name.toLowerCase(requestedLocale), name);
 140             }
 141         }
 142     }
 143 
 144     protected void registerFontsInDir(final String dirName, boolean useJavaRasterizer,
 145                                       int fontRank, boolean defer, boolean resolveSymLinks) {
 146 
 147         String[] files = AccessController.doPrivileged((PrivilegedAction<String[]>) () -> {
 148             return new File(dirName).list(getTrueTypeFilter());
 149         });
 150 
 151         if (files == null) {
 152            return;
 153         } else {
 154             for (String f : files) {
 155                 loadNativeDirFonts(dirName+File.separator+f);
 156             }
 157         }
 158         super.registerFontsInDir(dirName, useJavaRasterizer, fontRank, defer, resolveSymLinks);
 159     }
 160 
 161     private native void loadNativeDirFonts(String fontPath);
 162     private native void loadNativeFonts();
 163 
 164     void registerFont(String fontName, String fontFamilyName) {
 165         final CFont font = new CFont(fontName, fontFamilyName);
 166 
 167         registerGenericFont(font);
 168     }
 169 
 170     void registerItalicDerived() {
 171         FontFamily[] famArr = FontFamily.getAllFontFamilies();
 172         for (int i=0; i<famArr.length; i++) {
 173             FontFamily family = famArr[i];
 174 
 175             Font2D f2dPlain = family.getFont(Font.PLAIN);
 176             if (f2dPlain != null && !(f2dPlain instanceof CFont)) continue;
 177             Font2D f2dBold = family.getFont(Font.BOLD);
 178             if (f2dBold != null && !(f2dBold instanceof CFont)) continue;
 179             Font2D f2dItalic = family.getFont(Font.ITALIC);
 180             if (f2dItalic != null && !(f2dItalic instanceof CFont)) continue;
 181             Font2D f2dBoldItalic = family.getFont(Font.BOLD|Font.ITALIC);
 182             if (f2dBoldItalic != null && !(f2dBoldItalic instanceof CFont)) continue;
 183 
 184             CFont plain = (CFont)f2dPlain;
 185             CFont bold = (CFont)f2dBold;
 186             CFont italic = (CFont)f2dItalic;
 187             CFont boldItalic = (CFont)f2dBoldItalic;
 188 
 189             if (bold == null) bold = plain;
 190             if (plain == null && bold == null) continue;
 191             if (italic != null && boldItalic != null) continue;
 192             if (plain != null && italic == null) {
 193                registerGenericFont(plain.createItalicVariant(), true);
 194             }
 195             if (bold != null && boldItalic == null) {
 196                registerGenericFont(bold.createItalicVariant(), true);
 197             }
 198         }
 199     }
 200 
 201     Object waitForFontsToBeLoaded  = new Object();
 202     private boolean loadedAllFonts = false;
 203 
 204     public void loadFonts()
 205     {
 206         synchronized(waitForFontsToBeLoaded)
 207         {
 208             super.loadFonts();
 209             java.security.AccessController.doPrivileged(
 210                 new java.security.PrivilegedAction<Object>() {
 211                     public Object run() {
 212                         if (!loadedAllFonts) {
 213                            loadNativeFonts();
 214                            registerItalicDerived();
 215                            loadedAllFonts = true;
 216                         }
 217                         return null;
 218                     }
 219                 }
 220             );
 221 
 222             String defaultFont = "Lucida Grande";
 223             String defaultFallback = "Lucida Grande";
 224 
 225             setupLogicalFonts("Dialog", defaultFont, defaultFallback);
 226             setupLogicalFonts("Serif", "Times", "Times");
 227             setupLogicalFonts("SansSerif", defaultFont, defaultFallback);
 228             setupLogicalFonts("Monospaced", "Menlo", "Courier");
 229             setupLogicalFonts("DialogInput", defaultFont, defaultFallback);
 230         }
 231     }
 232 
 233     protected void setupLogicalFonts(String logicalName, String realName, String fallbackName) {
 234         FontFamily realFamily = getFontFamilyWithExtraTry(logicalName, realName, fallbackName);
 235 
 236         cloneStyledFont(realFamily, logicalName, Font.PLAIN);
 237         cloneStyledFont(realFamily, logicalName, Font.BOLD);
 238         cloneStyledFont(realFamily, logicalName, Font.ITALIC);
 239         cloneStyledFont(realFamily, logicalName, Font.BOLD | Font.ITALIC);
 240     }
 241 
 242     protected FontFamily getFontFamilyWithExtraTry(String logicalName, String realName, String fallbackName){
 243         FontFamily family = getFontFamily(realName, fallbackName);
 244         if (family != null) return family;
 245 
 246         // at this point, we recognize that we probably needed a fallback font
 247         super.loadFonts();
 248 
 249         family = getFontFamily(realName, fallbackName);
 250         if (family != null) return family;
 251 
 252         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.");
 253         return null;
 254     }
 255 
 256     protected FontFamily getFontFamily(String realName, String fallbackName){
 257         FontFamily family = FontFamily.getFamily(realName);
 258         if (family != null) return family;
 259 
 260         family = FontFamily.getFamily(fallbackName);
 261         if (family != null){
 262             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.");
 263             return family;
 264         }
 265 
 266         return null;
 267     }
 268 
 269     protected boolean cloneStyledFont(FontFamily realFamily, String logicalFamilyName, int style) {
 270         if (realFamily == null) return false;
 271 
 272         Font2D realFont = realFamily.getFontWithExactStyleMatch(style);
 273         if (realFont == null || !(realFont instanceof CFont)) return false;
 274 
 275         CFont newFont = new CFont((CFont)realFont, logicalFamilyName);
 276         registerGenericFont(newFont, true);
 277 
 278         return true;
 279     }
 280 
 281     @Override
 282     public String getFontPath(boolean noType1Fonts) {
 283         // In the case of the Cocoa toolkit, since we go through NSFont, we don't need to register /Library/Fonts
 284         Toolkit tk = Toolkit.getDefaultToolkit();
 285         if (tk instanceof HeadlessToolkit) {
 286             tk = ((HeadlessToolkit)tk).getUnderlyingToolkit();
 287         }
 288         if (tk instanceof LWCToolkit) {
 289             return "";
 290         }
 291 
 292         // X11 case
 293         return "/Library/Fonts";
 294     }
 295 
 296     @Override
 297     protected FontUIResource getFontConfigFUIR(
 298             String family, int style, int size)
 299     {
 300         String mappedName = FontUtilities.mapFcName(family);
 301         if (mappedName == null) {
 302             mappedName = "sansserif";
 303         }
 304         return new FontUIResource(mappedName, style, size);
 305     }
 306 
 307     // Only implemented on Windows
 308     @Override
 309     protected void populateFontFileNameMap(HashMap<String, String> fontToFileMap, HashMap<String, String> fontToFamilyNameMap,
 310             HashMap<String, ArrayList<String>> familyToFontListMap, Locale locale) {}
 311 }