6869705: Missing files of CR6795908, FontManager refactoring
Reviewed-by: prr, igor
This commit is contained in:
parent
eb12553aa3
commit
db5938b208
48
jdk/src/share/classes/sun/font/FontAccess.java
Normal file
48
jdk/src/share/classes/sun/font/FontAccess.java
Normal file
@ -0,0 +1,48 @@
|
||||
/*
|
||||
* Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
|
||||
* particular file as subject to the "Classpath" exception as provided
|
||||
* by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||||
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||||
* have any questions.
|
||||
*/
|
||||
|
||||
package sun.font;
|
||||
|
||||
import java.awt.Font;
|
||||
|
||||
public abstract class FontAccess {
|
||||
|
||||
private static FontAccess access;
|
||||
public static synchronized void setFontAccess(FontAccess acc) {
|
||||
if (access != null) {
|
||||
throw new InternalError("Attempt to set FontAccessor twice");
|
||||
}
|
||||
access = acc;
|
||||
}
|
||||
|
||||
public static synchronized FontAccess getFontAccess() {
|
||||
return access;
|
||||
}
|
||||
|
||||
public abstract Font2D getFont2D(Font f);
|
||||
public abstract void setFont2D(Font f, Font2DHandle h);
|
||||
public abstract void setCreatedFont(Font f);
|
||||
public abstract boolean isCreatedFont(Font f);
|
||||
}
|
106
jdk/src/share/classes/sun/font/FontManagerFactory.java
Normal file
106
jdk/src/share/classes/sun/font/FontManagerFactory.java
Normal file
@ -0,0 +1,106 @@
|
||||
/*
|
||||
* Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
|
||||
* particular file as subject to the "Classpath" exception as provided
|
||||
* by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||||
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||||
* have any questions.
|
||||
*/
|
||||
|
||||
package sun.font;
|
||||
|
||||
import java.awt.AWTError;
|
||||
import java.awt.Font;
|
||||
import java.awt.GraphicsEnvironment;
|
||||
import java.awt.Toolkit;
|
||||
import java.security.AccessController;
|
||||
import java.security.PrivilegedAction;
|
||||
|
||||
import sun.security.action.GetPropertyAction;
|
||||
|
||||
|
||||
/**
|
||||
* Factory class used to retrieve a valid FontManager instance for the current
|
||||
* platform.
|
||||
*
|
||||
* A default implementation is given for Linux, Solaris and Windows.
|
||||
* You can alter the behaviour of the {@link #getInstance()} method by setting
|
||||
* the {@code sun.font.fontmanager} property. For example:
|
||||
* {@code sun.font.fontmanager=sun.awt.X11FontManager}
|
||||
*/
|
||||
public final class FontManagerFactory {
|
||||
|
||||
/** Our singleton instance. */
|
||||
private static FontManager instance = null;
|
||||
|
||||
private static final String DEFAULT_CLASS;
|
||||
static {
|
||||
if (FontUtilities.isWindows)
|
||||
DEFAULT_CLASS = "sun.awt.Win32FontManager";
|
||||
else
|
||||
DEFAULT_CLASS = "sun.awt.X11FontManager";
|
||||
}
|
||||
|
||||
/**
|
||||
* Get a valid FontManager implementation for the current platform.
|
||||
*
|
||||
* @return a valid FontManager instance for the current platform
|
||||
*/
|
||||
public static synchronized FontManager getInstance() {
|
||||
|
||||
if (instance != null) {
|
||||
return instance;
|
||||
}
|
||||
|
||||
String fmClassName = AccessController.doPrivileged(
|
||||
new GetPropertyAction("sun.font.fontmanager",
|
||||
DEFAULT_CLASS));
|
||||
|
||||
try {
|
||||
@SuppressWarnings("unchecked")
|
||||
ClassLoader cl = (ClassLoader)
|
||||
AccessController.doPrivileged(new PrivilegedAction() {
|
||||
public Object run() {
|
||||
return ClassLoader.getSystemClassLoader();
|
||||
}
|
||||
});
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
Class fmClass = Class.forName(fmClassName, true, cl);
|
||||
instance = (FontManager) fmClass.newInstance();
|
||||
|
||||
} catch (ClassNotFoundException ex) {
|
||||
InternalError err = new InternalError();
|
||||
err.initCause(ex);
|
||||
throw err;
|
||||
|
||||
} catch (InstantiationException ex) {
|
||||
InternalError err = new InternalError();
|
||||
err.initCause(ex);
|
||||
throw err;
|
||||
|
||||
} catch (IllegalAccessException ex) {
|
||||
InternalError err = new InternalError();
|
||||
err.initCause(ex);
|
||||
throw err;
|
||||
}
|
||||
|
||||
return instance;
|
||||
}
|
||||
}
|
57
jdk/src/share/classes/sun/font/FontManagerForSGE.java
Normal file
57
jdk/src/share/classes/sun/font/FontManagerForSGE.java
Normal file
@ -0,0 +1,57 @@
|
||||
/*
|
||||
* Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
|
||||
* particular file as subject to the "Classpath" exception as provided
|
||||
* by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||||
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||||
* have any questions.
|
||||
*/
|
||||
|
||||
package sun.font;
|
||||
|
||||
import java.awt.Font;
|
||||
import java.util.Locale;
|
||||
import java.util.TreeMap;
|
||||
|
||||
/**
|
||||
* This is an extension of the {@link FontManager} interface which has to
|
||||
* be implemented on systems that want to use SunGraphicsEnvironment. It
|
||||
* adds a couple of methods that are only required by SGE. Graphics
|
||||
* implementations that use their own GraphicsEnvironment are not required
|
||||
* to implement this and can use plain FontManager instead.
|
||||
*/
|
||||
public interface FontManagerForSGE extends FontManager {
|
||||
|
||||
/**
|
||||
* Return an array of created Fonts, or null, if no fonts were created yet.
|
||||
*/
|
||||
public Font[] getCreatedFonts();
|
||||
|
||||
/**
|
||||
* Similar to getCreatedFonts, but returns a TreeMap of fonts by family name.
|
||||
*/
|
||||
public TreeMap<String, String> getCreatedFontFamilyNames();
|
||||
|
||||
/**
|
||||
* Returns all fonts installed in this environment.
|
||||
*/
|
||||
public Font[] getAllInstalledFonts();
|
||||
|
||||
public String[] getInstalledFontFamilyNames(Locale requestedLocale);
|
||||
}
|
486
jdk/src/share/classes/sun/font/FontUtilities.java
Normal file
486
jdk/src/share/classes/sun/font/FontUtilities.java
Normal file
@ -0,0 +1,486 @@
|
||||
/*
|
||||
* Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
|
||||
* particular file as subject to the "Classpath" exception as provided
|
||||
* by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||||
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||||
* have any questions.
|
||||
*/
|
||||
|
||||
package sun.font;
|
||||
|
||||
import java.awt.Font;
|
||||
import java.io.BufferedReader;
|
||||
import java.io.File;
|
||||
import java.io.FileInputStream;
|
||||
import java.io.InputStreamReader;
|
||||
import java.security.AccessController;
|
||||
import java.util.logging.Level;
|
||||
import java.util.logging.Logger;
|
||||
|
||||
import javax.swing.plaf.FontUIResource;
|
||||
|
||||
import sun.security.action.GetPropertyAction;
|
||||
|
||||
/**
|
||||
* A collection of utility methods.
|
||||
*/
|
||||
public final class FontUtilities {
|
||||
|
||||
public static final boolean isSolaris;
|
||||
|
||||
public static final boolean isLinux;
|
||||
|
||||
public static final boolean isSolaris8;
|
||||
|
||||
public static final boolean isSolaris9;
|
||||
|
||||
public static final boolean isOpenSolaris;
|
||||
|
||||
public static final boolean useT2K;
|
||||
|
||||
public static final boolean isWindows;
|
||||
|
||||
public static final boolean isOpenJDK;
|
||||
|
||||
static final String LUCIDA_FILE_NAME = "LucidaSansRegular.ttf";
|
||||
|
||||
// This static initializer block figures out the OS constants.
|
||||
static {
|
||||
|
||||
String osName = AccessController.doPrivileged(
|
||||
new GetPropertyAction("os.name", "unknownOS"));
|
||||
isSolaris = osName.startsWith("SunOS");
|
||||
|
||||
isLinux = osName.startsWith("Linux");
|
||||
|
||||
String t2kStr = AccessController.doPrivileged(
|
||||
new GetPropertyAction("sun.java2d.font.scaler"));
|
||||
if (t2kStr != null) {
|
||||
useT2K = "t2k".equals(t2kStr);
|
||||
} else {
|
||||
useT2K = false;
|
||||
}
|
||||
if (isSolaris) {
|
||||
String version = AccessController.doPrivileged(
|
||||
new GetPropertyAction("os.version", "0.0"));
|
||||
isSolaris8 = version.startsWith("5.8");
|
||||
isSolaris9 = version.startsWith("5.9");
|
||||
float ver = Float.parseFloat(version);
|
||||
if (ver > 5.10f) {
|
||||
File f = new File("/etc/release");
|
||||
String line = null;
|
||||
try {
|
||||
FileInputStream fis = new FileInputStream(f);
|
||||
InputStreamReader isr = new InputStreamReader(
|
||||
fis, "ISO-8859-1");
|
||||
BufferedReader br = new BufferedReader(isr);
|
||||
line = br.readLine();
|
||||
fis.close();
|
||||
} catch (Exception ex) {
|
||||
// Nothing to do here.
|
||||
}
|
||||
if (line != null && line.indexOf("OpenSolaris") >= 0) {
|
||||
isOpenSolaris = true;
|
||||
} else {
|
||||
isOpenSolaris = false;
|
||||
}
|
||||
} else {
|
||||
isOpenSolaris= false;
|
||||
}
|
||||
} else {
|
||||
isSolaris8 = false;
|
||||
isSolaris9 = false;
|
||||
isOpenSolaris = false;
|
||||
}
|
||||
isWindows = osName.startsWith("Windows");
|
||||
String jreLibDirName = AccessController.doPrivileged(
|
||||
new GetPropertyAction("java.home","")) + File.separator + "lib";
|
||||
String jreFontDirName = jreLibDirName + File.separator + "fonts";
|
||||
File lucidaFile =
|
||||
new File(jreFontDirName + File.separator + LUCIDA_FILE_NAME);
|
||||
isOpenJDK = !lucidaFile.exists();
|
||||
}
|
||||
|
||||
/**
|
||||
* Referenced by code in the JDK which wants to test for the
|
||||
* minimum char code for which layout may be required.
|
||||
* Note that even basic latin text can benefit from ligatures,
|
||||
* eg "ffi" but we presently apply those only if explicitly
|
||||
* requested with TextAttribute.LIGATURES_ON.
|
||||
* The value here indicates the lowest char code for which failing
|
||||
* to invoke layout would prevent acceptable rendering.
|
||||
*/
|
||||
public static final int MIN_LAYOUT_CHARCODE = 0x0300;
|
||||
|
||||
/**
|
||||
* Referenced by code in the JDK which wants to test for the
|
||||
* maximum char code for which layout may be required.
|
||||
* Note this does not account for supplementary characters
|
||||
* where the caller interprets 'layout' to mean any case where
|
||||
* one 'char' (ie the java type char) does not map to one glyph
|
||||
*/
|
||||
public static final int MAX_LAYOUT_CHARCODE = 0x206F;
|
||||
|
||||
private static boolean debugFonts = false;
|
||||
private static Logger logger = null;
|
||||
private static boolean logging;
|
||||
|
||||
static {
|
||||
|
||||
String debugLevel =
|
||||
System.getProperty("sun.java2d.debugfonts");
|
||||
|
||||
if (debugLevel != null && !debugLevel.equals("false")) {
|
||||
debugFonts = true;
|
||||
logger = Logger.getLogger("sun.java2d");
|
||||
if (debugLevel.equals("warning")) {
|
||||
logger.setLevel(Level.WARNING);
|
||||
} else if (debugLevel.equals("severe")) {
|
||||
logger.setLevel(Level.SEVERE);
|
||||
}
|
||||
}
|
||||
|
||||
if (debugFonts) {
|
||||
logger = Logger.getLogger("sun.java2d", null);
|
||||
logging = logger.getLevel() != Level.OFF;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* Calls the private getFont2D() method in java.awt.Font objects.
|
||||
*
|
||||
* @param font the font object to call
|
||||
*
|
||||
* @return the Font2D object returned by Font.getFont2D()
|
||||
*/
|
||||
public static Font2D getFont2D(Font font) {
|
||||
return FontAccess.getFontAccess().getFont2D(font);
|
||||
}
|
||||
|
||||
/**
|
||||
* If there is anything in the text which triggers a case
|
||||
* where char->glyph does not map 1:1 in straightforward
|
||||
* left->right ordering, then this method returns true.
|
||||
* Scripts which might require it but are not treated as such
|
||||
* due to JDK implementations will not return true.
|
||||
* ie a 'true' return is an indication of the treatment by
|
||||
* the implementation.
|
||||
* Whether supplementary characters should be considered is dependent
|
||||
* on the needs of the caller. Since this method accepts the 'char' type
|
||||
* then such chars are always represented by a pair. From a rendering
|
||||
* perspective these will all (in the cases I know of) still be one
|
||||
* unicode character -> one glyph. But if a caller is using this to
|
||||
* discover any case where it cannot make naive assumptions about
|
||||
* the number of chars, and how to index through them, then it may
|
||||
* need the option to have a 'true' return in such a case.
|
||||
*/
|
||||
public static boolean isComplexText(char [] chs, int start, int limit) {
|
||||
|
||||
for (int i = start; i < limit; i++) {
|
||||
if (chs[i] < MIN_LAYOUT_CHARCODE) {
|
||||
continue;
|
||||
}
|
||||
else if (isNonSimpleChar(chs[i])) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/* This is almost the same as the method above, except it takes a
|
||||
* char which means it may include undecoded surrogate pairs.
|
||||
* The distinction is made so that code which needs to identify all
|
||||
* cases in which we do not have a simple mapping from
|
||||
* char->unicode character->glyph can be be identified.
|
||||
* For example measurement cannot simply sum advances of 'chars',
|
||||
* the caret in editable text cannot advance one 'char' at a time, etc.
|
||||
* These callers really are asking for more than whether 'layout'
|
||||
* needs to be run, they need to know if they can assume 1->1
|
||||
* char->glyph mapping.
|
||||
*/
|
||||
public static boolean isNonSimpleChar(char ch) {
|
||||
return
|
||||
isComplexCharCode(ch) ||
|
||||
(ch >= CharToGlyphMapper.HI_SURROGATE_START &&
|
||||
ch <= CharToGlyphMapper.LO_SURROGATE_END);
|
||||
}
|
||||
|
||||
/* If the character code falls into any of a number of unicode ranges
|
||||
* where we know that simple left->right layout mapping chars to glyphs
|
||||
* 1:1 and accumulating advances is going to produce incorrect results,
|
||||
* we want to know this so the caller can use a more intelligent layout
|
||||
* approach. A caller who cares about optimum performance may want to
|
||||
* check the first case and skip the method call if its in that range.
|
||||
* Although there's a lot of tests in here, knowing you can skip
|
||||
* CTL saves a great deal more. The rest of the checks are ordered
|
||||
* so that rather than checking explicitly if (>= start & <= end)
|
||||
* which would mean all ranges would need to be checked so be sure
|
||||
* CTL is not needed, the method returns as soon as it recognises
|
||||
* the code point is outside of a CTL ranges.
|
||||
* NOTE: Since this method accepts an 'int' it is asssumed to properly
|
||||
* represent a CHARACTER. ie it assumes the caller has already
|
||||
* converted surrogate pairs into supplementary characters, and so
|
||||
* can handle this case and doesn't need to be told such a case is
|
||||
* 'complex'.
|
||||
*/
|
||||
public static boolean isComplexCharCode(int code) {
|
||||
|
||||
if (code < MIN_LAYOUT_CHARCODE || code > MAX_LAYOUT_CHARCODE) {
|
||||
return false;
|
||||
}
|
||||
else if (code <= 0x036f) {
|
||||
// Trigger layout for combining diacriticals 0x0300->0x036f
|
||||
return true;
|
||||
}
|
||||
else if (code < 0x0590) {
|
||||
// No automatic layout for Greek, Cyrillic, Armenian.
|
||||
return false;
|
||||
}
|
||||
else if (code <= 0x06ff) {
|
||||
// Hebrew 0590 - 05ff
|
||||
// Arabic 0600 - 06ff
|
||||
return true;
|
||||
}
|
||||
else if (code < 0x0900) {
|
||||
return false; // Syriac and Thaana
|
||||
}
|
||||
else if (code <= 0x0e7f) {
|
||||
// if Indic, assume shaping for conjuncts, reordering:
|
||||
// 0900 - 097F Devanagari
|
||||
// 0980 - 09FF Bengali
|
||||
// 0A00 - 0A7F Gurmukhi
|
||||
// 0A80 - 0AFF Gujarati
|
||||
// 0B00 - 0B7F Oriya
|
||||
// 0B80 - 0BFF Tamil
|
||||
// 0C00 - 0C7F Telugu
|
||||
// 0C80 - 0CFF Kannada
|
||||
// 0D00 - 0D7F Malayalam
|
||||
// 0D80 - 0DFF Sinhala
|
||||
// 0E00 - 0E7F if Thai, assume shaping for vowel, tone marks
|
||||
return true;
|
||||
}
|
||||
else if (code < 0x1780) {
|
||||
return false;
|
||||
}
|
||||
else if (code <= 0x17ff) { // 1780 - 17FF Khmer
|
||||
return true;
|
||||
}
|
||||
else if (code < 0x200c) {
|
||||
return false;
|
||||
}
|
||||
else if (code <= 0x200d) { // zwj or zwnj
|
||||
return true;
|
||||
}
|
||||
else if (code >= 0x202a && code <= 0x202e) { // directional control
|
||||
return true;
|
||||
}
|
||||
else if (code >= 0x206a && code <= 0x206f) { // directional control
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
public static Logger getLogger() {
|
||||
return logger;
|
||||
}
|
||||
|
||||
public static boolean isLogging() {
|
||||
return logging;
|
||||
}
|
||||
|
||||
public static boolean debugFonts() {
|
||||
return debugFonts;
|
||||
}
|
||||
|
||||
|
||||
// The following methods are used by Swing.
|
||||
|
||||
/* Revise the implementation to in fact mean "font is a composite font.
|
||||
* This ensures that Swing components will always benefit from the
|
||||
* fall back fonts
|
||||
*/
|
||||
public static boolean fontSupportsDefaultEncoding(Font font) {
|
||||
return getFont2D(font) instanceof CompositeFont;
|
||||
}
|
||||
|
||||
/**
|
||||
* This method is provided for internal and exclusive use by Swing.
|
||||
*
|
||||
* It may be used in conjunction with fontSupportsDefaultEncoding(Font)
|
||||
* In the event that a desktop properties font doesn't directly
|
||||
* support the default encoding, (ie because the host OS supports
|
||||
* adding support for the current locale automatically for native apps),
|
||||
* then Swing calls this method to get a font which uses the specified
|
||||
* font for the code points it covers, but also supports this locale
|
||||
* just as the standard composite fonts do.
|
||||
* Note: this will over-ride any setting where an application
|
||||
* specifies it prefers locale specific composite fonts.
|
||||
* The logic for this, is that this method is used only where the user or
|
||||
* application has specified that the native L&F be used, and that
|
||||
* we should honour that request to use the same font as native apps use.
|
||||
*
|
||||
* The behaviour of this method is to construct a new composite
|
||||
* Font object that uses the specified physical font as its first
|
||||
* component, and adds all the components of "dialog" as fall back
|
||||
* components.
|
||||
* The method currently assumes that only the size and style attributes
|
||||
* are set on the specified font. It doesn't copy the font transform or
|
||||
* other attributes because they aren't set on a font created from
|
||||
* the desktop. This will need to be fixed if use is broadened.
|
||||
*
|
||||
* Operations such as Font.deriveFont will work properly on the
|
||||
* font returned by this method for deriving a different point size.
|
||||
* Additionally it tries to support a different style by calling
|
||||
* getNewComposite() below. That also supports replacing slot zero
|
||||
* with a different physical font but that is expected to be "rare".
|
||||
* Deriving with a different style is needed because its been shown
|
||||
* that some applications try to do this for Swing FontUIResources.
|
||||
* Also operations such as new Font(font.getFontName(..), Font.PLAIN, 14);
|
||||
* will NOT yield the same result, as the new underlying CompositeFont
|
||||
* cannot be "looked up" in the font registry.
|
||||
* This returns a FontUIResource as that is the Font sub-class needed
|
||||
* by Swing.
|
||||
* Suggested usage is something like :
|
||||
* FontUIResource fuir;
|
||||
* Font desktopFont = getDesktopFont(..);
|
||||
* // NOTE even if fontSupportsDefaultEncoding returns true because
|
||||
* // you get Tahoma and are running in an English locale, you may
|
||||
* // still want to just call getCompositeFontUIResource() anyway
|
||||
* // as only then will you get fallback fonts - eg for CJK.
|
||||
* if (FontManager.fontSupportsDefaultEncoding(desktopFont)) {
|
||||
* fuir = new FontUIResource(..);
|
||||
* } else {
|
||||
* fuir = FontManager.getCompositeFontUIResource(desktopFont);
|
||||
* }
|
||||
* return fuir;
|
||||
*/
|
||||
public static FontUIResource getCompositeFontUIResource(Font font) {
|
||||
|
||||
FontUIResource fuir =
|
||||
new FontUIResource(font.getName(),font.getStyle(),font.getSize());
|
||||
Font2D font2D = FontUtilities.getFont2D(font);
|
||||
|
||||
if (!(font2D instanceof PhysicalFont)) {
|
||||
/* Swing should only be calling this when a font is obtained
|
||||
* from desktop properties, so should generally be a physical font,
|
||||
* an exception might be for names like "MS Serif" which are
|
||||
* automatically mapped to "Serif", so there's no need to do
|
||||
* anything special in that case. But note that suggested usage
|
||||
* is first to call fontSupportsDefaultEncoding(Font) and this
|
||||
* method should not be called if that were to return true.
|
||||
*/
|
||||
return fuir;
|
||||
}
|
||||
|
||||
FontManager fm = FontManagerFactory.getInstance();
|
||||
CompositeFont dialog2D =
|
||||
(CompositeFont) fm.findFont2D("dialog", font.getStyle(), FontManager.NO_FALLBACK);
|
||||
if (dialog2D == null) { /* shouldn't happen */
|
||||
return fuir;
|
||||
}
|
||||
PhysicalFont physicalFont = (PhysicalFont)font2D;
|
||||
CompositeFont compFont = new CompositeFont(physicalFont, dialog2D);
|
||||
FontAccess.getFontAccess().setFont2D(fuir, compFont.handle);
|
||||
/* marking this as a created font is needed as only created fonts
|
||||
* copy their creator's handles.
|
||||
*/
|
||||
FontAccess.getFontAccess().setCreatedFont(fuir);
|
||||
return fuir;
|
||||
}
|
||||
|
||||
/* A small "map" from GTK/fontconfig names to the equivalent JDK
|
||||
* logical font name.
|
||||
*/
|
||||
private static final String[][] nameMap = {
|
||||
{"sans", "sansserif"},
|
||||
{"sans-serif", "sansserif"},
|
||||
{"serif", "serif"},
|
||||
{"monospace", "monospaced"}
|
||||
};
|
||||
|
||||
public static String mapFcName(String name) {
|
||||
for (int i = 0; i < nameMap.length; i++) {
|
||||
if (name.equals(nameMap[i][0])) {
|
||||
return nameMap[i][1];
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
|
||||
/* This is called by Swing passing in a fontconfig family name
|
||||
* such as "sans". In return Swing gets a FontUIResource instance
|
||||
* that has queried fontconfig to resolve the font(s) used for this.
|
||||
* Fontconfig will if asked return a list of fonts to give the largest
|
||||
* possible code point coverage.
|
||||
* For now we use only the first font returned by fontconfig, and
|
||||
* back it up with the most closely matching JDK logical font.
|
||||
* Essentially this means pre-pending what we return now with fontconfig's
|
||||
* preferred physical font. This could lead to some duplication in cases,
|
||||
* if we already included that font later. We probably should remove such
|
||||
* duplicates, but it is not a significant problem. It can be addressed
|
||||
* later as part of creating a Composite which uses more of the
|
||||
* same fonts as fontconfig. At that time we also should pay more
|
||||
* attention to the special rendering instructions fontconfig returns,
|
||||
* such as whether we should prefer embedded bitmaps over antialiasing.
|
||||
* There's no way to express that via a Font at present.
|
||||
*/
|
||||
public static FontUIResource getFontConfigFUIR(String fcFamily,
|
||||
int style, int size) {
|
||||
|
||||
String mapped = mapFcName(fcFamily);
|
||||
if (mapped == null) {
|
||||
mapped = "sansserif";
|
||||
}
|
||||
|
||||
FontUIResource fuir;
|
||||
FontManager fm = FontManagerFactory.getInstance();
|
||||
if (fm instanceof SunFontManager) {
|
||||
SunFontManager sfm = (SunFontManager) fm;
|
||||
fuir = sfm.getFontConfigFUIR(mapped, style, size);
|
||||
} else {
|
||||
fuir = new FontUIResource(mapped, style, size);
|
||||
}
|
||||
return fuir;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Used by windows printing to assess if a font is likely to
|
||||
* be layout compatible with JDK
|
||||
* TrueType fonts should be, but if they have no GPOS table,
|
||||
* but do have a GSUB table, then they are probably older
|
||||
* fonts GDI handles differently.
|
||||
*/
|
||||
public static boolean textLayoutIsCompatible(Font font) {
|
||||
|
||||
Font2D font2D = getFont2D(font);
|
||||
if (font2D instanceof TrueTypeFont) {
|
||||
TrueTypeFont ttf = (TrueTypeFont) font2D;
|
||||
return
|
||||
ttf.getDirectoryEntry(TrueTypeFont.GSUBTag) == null ||
|
||||
ttf.getDirectoryEntry(TrueTypeFont.GPOSTag) != null;
|
||||
} else {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
}
|
3675
jdk/src/share/classes/sun/font/SunFontManager.java
Normal file
3675
jdk/src/share/classes/sun/font/SunFontManager.java
Normal file
File diff suppressed because it is too large
Load Diff
850
jdk/src/solaris/classes/sun/awt/X11FontManager.java
Normal file
850
jdk/src/solaris/classes/sun/awt/X11FontManager.java
Normal file
@ -0,0 +1,850 @@
|
||||
package sun.awt;
|
||||
|
||||
import java.awt.GraphicsEnvironment;
|
||||
import java.io.BufferedReader;
|
||||
import java.io.File;
|
||||
import java.io.FileReader;
|
||||
import java.io.IOException;
|
||||
import java.io.StreamTokenizer;
|
||||
import java.util.HashMap;
|
||||
import java.util.HashSet;
|
||||
import java.util.Locale;
|
||||
import java.util.Map;
|
||||
import java.util.NoSuchElementException;
|
||||
import java.util.StringTokenizer;
|
||||
import java.util.Vector;
|
||||
import java.util.logging.Logger;
|
||||
|
||||
import javax.swing.plaf.FontUIResource;
|
||||
import sun.awt.motif.MFontConfiguration;
|
||||
import sun.font.CompositeFont;
|
||||
import sun.font.FontManager;
|
||||
import sun.font.SunFontManager;
|
||||
import sun.font.FontConfigManager;
|
||||
import sun.font.FcFontConfiguration;
|
||||
import sun.font.FontAccess;
|
||||
import sun.font.FontUtilities;
|
||||
import sun.font.NativeFont;
|
||||
|
||||
/**
|
||||
* The X11 implementation of {@link FontManager}.
|
||||
*/
|
||||
public class X11FontManager extends SunFontManager {
|
||||
|
||||
// constants identifying XLFD and font ID fields
|
||||
private static final int FOUNDRY_FIELD = 1;
|
||||
private static final int FAMILY_NAME_FIELD = 2;
|
||||
private static final int WEIGHT_NAME_FIELD = 3;
|
||||
private static final int SLANT_FIELD = 4;
|
||||
private static final int SETWIDTH_NAME_FIELD = 5;
|
||||
private static final int ADD_STYLE_NAME_FIELD = 6;
|
||||
private static final int PIXEL_SIZE_FIELD = 7;
|
||||
private static final int POINT_SIZE_FIELD = 8;
|
||||
private static final int RESOLUTION_X_FIELD = 9;
|
||||
private static final int RESOLUTION_Y_FIELD = 10;
|
||||
private static final int SPACING_FIELD = 11;
|
||||
private static final int AVERAGE_WIDTH_FIELD = 12;
|
||||
private static final int CHARSET_REGISTRY_FIELD = 13;
|
||||
private static final int CHARSET_ENCODING_FIELD = 14;
|
||||
|
||||
/*
|
||||
* fontNameMap is a map from a fontID (which is a substring of an XLFD like
|
||||
* "-monotype-arial-bold-r-normal-iso8859-7")
|
||||
* to font file path like
|
||||
* /usr/openwin/lib/locale/iso_8859_7/X11/fonts/TrueType/ArialBoldItalic.ttf
|
||||
* It's used in a couple of methods like
|
||||
* getFileNameFomPlatformName(..) to help locate the font file.
|
||||
* We use this substring of a full XLFD because the font configuration files
|
||||
* define the XLFDs in a way that's easier to make into a request.
|
||||
* E.g., the -0-0-0-0-p-0- reported by X is -*-%d-*-*-p-*- in the font
|
||||
* configuration files. We need to remove that part for comparisons.
|
||||
*/
|
||||
private static Map fontNameMap = new HashMap();
|
||||
|
||||
/*
|
||||
* xlfdMap is a map from a platform path like
|
||||
* /usr/openwin/lib/locale/ja/X11/fonts/TT/HG-GothicB.ttf to an XLFD like
|
||||
* "-ricoh-hg gothic b-medium-r-normal--0-0-0-0-m-0-jisx0201.1976-0"
|
||||
* Because there may be multiple native names, because the font is used
|
||||
* to support multiple X encodings for example, the value of an entry in
|
||||
* this map is always a vector where we store all the native names.
|
||||
* For fonts which we don't understand the key isn't a pathname, its
|
||||
* the full XLFD string like :-
|
||||
* "-ricoh-hg gothic b-medium-r-normal--0-0-0-0-m-0-jisx0201.1976-0"
|
||||
*/
|
||||
private static Map xlfdMap = new HashMap();
|
||||
|
||||
/* xFontDirsMap is also a map from a font ID to a font filepath.
|
||||
* The difference from fontNameMap is just that it does not have
|
||||
* resolved symbolic links. Normally this is not interesting except
|
||||
* that we need to know the directory in which a font was found to
|
||||
* add it to the X font server path, since although the files may
|
||||
* be linked, the fonts.dir is different and specific to the encoding
|
||||
* handled by that directory. This map is nulled out after use to free
|
||||
* heap space. If the optimal path is taken, such that all fonts in
|
||||
* font configuration files are referenced by filename, then the font
|
||||
* dir can be directly derived as its parent directory.
|
||||
* If a font is used by two XLFDs, each corresponding to a different
|
||||
* X11 font directory, then precautions must be taken to include both
|
||||
* directories.
|
||||
*/
|
||||
private static Map xFontDirsMap;
|
||||
|
||||
/*
|
||||
* This is the set of font directories needed to be on the X font path
|
||||
* to enable AWT heavyweights to find all of the font configuration fonts.
|
||||
* It is populated by :
|
||||
* - awtfontpath entries in the fontconfig.properties
|
||||
* - parent directories of "core" fonts used in the fontconfig.properties
|
||||
* - looking up font dirs in the xFontDirsMap where the key is a fontID
|
||||
* (cut down version of the XLFD read from the font configuration file).
|
||||
* This set is nulled out after use to free heap space.
|
||||
*/
|
||||
private static HashSet<String> fontConfigDirs = null;
|
||||
|
||||
/* These maps are used on Linux where we reference the Lucida oblique
|
||||
* fonts in fontconfig files even though they aren't in the standard
|
||||
* font directory. This explicitly remaps the XLFDs for these to the
|
||||
* correct base font. This is needed to prevent composite fonts from
|
||||
* defaulting to the Lucida Sans which is a bad substitute for the
|
||||
* monospaced Lucida Sans Typewriter. Also these maps prevent the
|
||||
* JRE from doing wasted work at start up.
|
||||
*/
|
||||
HashMap<String, String> oblmap = null;
|
||||
|
||||
|
||||
/*
|
||||
* Used to eliminate redundant work. When a font directory is
|
||||
* registered it added to this list. Subsequent registrations for the
|
||||
* same directory can then be skipped by checking this Map.
|
||||
* Access to this map is not synchronised here since creation
|
||||
* of the singleton GE instance is already synchronised and that is
|
||||
* the only code path that accesses this map.
|
||||
*/
|
||||
private static HashMap registeredDirs = new HashMap();
|
||||
|
||||
/* Array of directories to be added to the X11 font path.
|
||||
* Used by static method called from Toolkits which use X11 fonts.
|
||||
* Specifically this means MToolkit
|
||||
*/
|
||||
private static String[] fontdirs = null;
|
||||
|
||||
private static String[] defaultPlatformFont = null;
|
||||
|
||||
private FontConfigManager fcManager = null;
|
||||
|
||||
public static X11FontManager getInstance() {
|
||||
return (X11FontManager) SunFontManager.getInstance();
|
||||
}
|
||||
|
||||
/**
|
||||
* Takes family name property in the following format:
|
||||
* "-linotype-helvetica-medium-r-normal-sans-*-%d-*-*-p-*-iso8859-1"
|
||||
* and returns the name of the corresponding physical font.
|
||||
* This code is used to resolve font configuration fonts, and expects
|
||||
* only to get called for these fonts.
|
||||
*/
|
||||
@Override
|
||||
public String getFileNameFromPlatformName(String platName) {
|
||||
|
||||
/* If the FontConfig file doesn't use xlfds, or its
|
||||
* FcFontConfiguration, this may be already a file name.
|
||||
*/
|
||||
if (platName.startsWith("/")) {
|
||||
return platName;
|
||||
}
|
||||
|
||||
String fileName = null;
|
||||
String fontID = specificFontIDForName(platName);
|
||||
|
||||
/* If the font filename has been explicitly assigned in the
|
||||
* font configuration file, use it. This avoids accessing
|
||||
* the wrong fonts on Linux, where different fonts (some
|
||||
* of which may not be usable by 2D) may share the same
|
||||
* specific font ID. It may also speed up the lookup.
|
||||
*/
|
||||
fileName = super.getFileNameFromPlatformName(platName);
|
||||
if (fileName != null) {
|
||||
if (isHeadless() && fileName.startsWith("-")) {
|
||||
/* if it's headless, no xlfd should be used */
|
||||
return null;
|
||||
}
|
||||
if (fileName.startsWith("/")) {
|
||||
/* If a path is assigned in the font configuration file,
|
||||
* it is required that the config file also specify using the
|
||||
* new awtfontpath key the X11 font directories
|
||||
* which must be added to the X11 font path to support
|
||||
* AWT access to that font. For that reason we no longer
|
||||
* have code here to add the parent directory to the list
|
||||
* of font config dirs, since the parent directory may not
|
||||
* be sufficient if fonts are symbolically linked to a
|
||||
* different directory.
|
||||
*
|
||||
* Add this XLFD (platform name) to the list of known
|
||||
* ones for this file.
|
||||
*/
|
||||
Vector xVal = (Vector) xlfdMap.get(fileName);
|
||||
if (xVal == null) {
|
||||
/* Try to be robust on Linux distros which move fonts
|
||||
* around by verifying that the fileName represents a
|
||||
* file that exists. If it doesn't, set it to null
|
||||
* to trigger a search.
|
||||
*/
|
||||
if (getFontConfiguration().needToSearchForFile(fileName)) {
|
||||
fileName = null;
|
||||
}
|
||||
if (fileName != null) {
|
||||
xVal = new Vector();
|
||||
xVal.add(platName);
|
||||
xlfdMap.put(fileName, xVal);
|
||||
}
|
||||
} else {
|
||||
if (!xVal.contains(platName)) {
|
||||
xVal.add(platName);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (fileName != null) {
|
||||
fontNameMap.put(fontID, fileName);
|
||||
return fileName;
|
||||
}
|
||||
}
|
||||
|
||||
if (fontID != null) {
|
||||
fileName = (String)fontNameMap.get(fontID);
|
||||
/* On Linux check for the Lucida Oblique fonts */
|
||||
if (fileName == null && FontUtilities.isLinux && !isOpenJDK()) {
|
||||
if (oblmap == null) {
|
||||
initObliqueLucidaFontMap();
|
||||
}
|
||||
String oblkey = getObliqueLucidaFontID(fontID);
|
||||
if (oblkey != null) {
|
||||
fileName = oblmap.get(oblkey);
|
||||
}
|
||||
}
|
||||
if (fontPath == null &&
|
||||
(fileName == null || !fileName.startsWith("/"))) {
|
||||
if (FontUtilities.debugFonts()) {
|
||||
FontUtilities.getLogger()
|
||||
.warning("** Registering all font paths because " +
|
||||
"can't find file for " + platName);
|
||||
}
|
||||
fontPath = getPlatformFontPath(noType1Font);
|
||||
registerFontDirs(fontPath);
|
||||
if (FontUtilities.debugFonts()) {
|
||||
FontUtilities.getLogger()
|
||||
.warning("** Finished registering all font paths");
|
||||
}
|
||||
fileName = (String)fontNameMap.get(fontID);
|
||||
}
|
||||
if (fileName == null && !isHeadless()) {
|
||||
/* Query X11 directly to see if this font is available
|
||||
* as a native font.
|
||||
*/
|
||||
fileName = getX11FontName(platName);
|
||||
}
|
||||
if (fileName == null) {
|
||||
fontID = switchFontIDForName(platName);
|
||||
fileName = (String)fontNameMap.get(fontID);
|
||||
}
|
||||
if (fileName != null) {
|
||||
fontNameMap.put(fontID, fileName);
|
||||
}
|
||||
}
|
||||
return fileName;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected String[] getNativeNames(String fontFileName,
|
||||
String platformName) {
|
||||
Vector nativeNames;
|
||||
if ((nativeNames=(Vector)xlfdMap.get(fontFileName))==null) {
|
||||
if (platformName == null) {
|
||||
return null;
|
||||
} else {
|
||||
/* back-stop so that at least the name used in the
|
||||
* font configuration file is known as a native name
|
||||
*/
|
||||
String []natNames = new String[1];
|
||||
natNames[0] = platformName;
|
||||
return natNames;
|
||||
}
|
||||
} else {
|
||||
int len = nativeNames.size();
|
||||
return (String[])nativeNames.toArray(new String[len]);
|
||||
}
|
||||
}
|
||||
|
||||
/* NOTE: this method needs to be executed in a privileged context.
|
||||
* The superclass constructor which is the primary caller of
|
||||
* this method executes entirely in such a context. Additionally
|
||||
* the loadFonts() method does too. So all should be well.
|
||||
|
||||
*/
|
||||
@Override
|
||||
protected void registerFontDir(String path) {
|
||||
/* fonts.dir file format looks like :-
|
||||
* 47
|
||||
* Arial.ttf -monotype-arial-regular-r-normal--0-0-0-0-p-0-iso8859-1
|
||||
* Arial-Bold.ttf -monotype-arial-bold-r-normal--0-0-0-0-p-0-iso8859-1
|
||||
* ...
|
||||
*/
|
||||
if (FontUtilities.debugFonts()) {
|
||||
FontUtilities.getLogger().info("ParseFontDir " + path);
|
||||
}
|
||||
File fontsDotDir = new File(path + File.separator + "fonts.dir");
|
||||
FileReader fr = null;
|
||||
try {
|
||||
if (fontsDotDir.canRead()) {
|
||||
fr = new FileReader(fontsDotDir);
|
||||
BufferedReader br = new BufferedReader(fr, 8192);
|
||||
StreamTokenizer st = new StreamTokenizer(br);
|
||||
st.eolIsSignificant(true);
|
||||
int ttype = st.nextToken();
|
||||
if (ttype == StreamTokenizer.TT_NUMBER) {
|
||||
int numEntries = (int)st.nval;
|
||||
ttype = st.nextToken();
|
||||
if (ttype == StreamTokenizer.TT_EOL) {
|
||||
st.resetSyntax();
|
||||
st.wordChars(32, 127);
|
||||
st.wordChars(128 + 32, 255);
|
||||
st.whitespaceChars(0, 31);
|
||||
|
||||
for (int i=0; i < numEntries; i++) {
|
||||
ttype = st.nextToken();
|
||||
if (ttype == StreamTokenizer.TT_EOF) {
|
||||
break;
|
||||
}
|
||||
if (ttype != StreamTokenizer.TT_WORD) {
|
||||
break;
|
||||
}
|
||||
int breakPos = st.sval.indexOf(' ');
|
||||
if (breakPos <= 0) {
|
||||
/* On TurboLinux 8.0 a fonts.dir file had
|
||||
* a line with integer value "24" which
|
||||
* appeared to be the number of remaining
|
||||
* entries in the file. This didn't add to
|
||||
* the value on the first line of the file.
|
||||
* Seemed like XFree86 didn't like this line
|
||||
* much either. It failed to parse the file.
|
||||
* Ignore lines like this completely, and
|
||||
* don't let them count as an entry.
|
||||
*/
|
||||
numEntries++;
|
||||
ttype = st.nextToken();
|
||||
if (ttype != StreamTokenizer.TT_EOL) {
|
||||
break;
|
||||
}
|
||||
|
||||
continue;
|
||||
}
|
||||
if (st.sval.charAt(0) == '!') {
|
||||
/* TurboLinux 8.0 comment line: ignore.
|
||||
* can't use st.commentChar('!') to just
|
||||
* skip because this line mustn't count
|
||||
* against numEntries.
|
||||
*/
|
||||
numEntries++;
|
||||
ttype = st.nextToken();
|
||||
if (ttype != StreamTokenizer.TT_EOL) {
|
||||
break;
|
||||
}
|
||||
continue;
|
||||
}
|
||||
String fileName = st.sval.substring(0, breakPos);
|
||||
/* TurboLinux 8.0 uses some additional syntax to
|
||||
* indicate algorithmic styling values.
|
||||
* Ignore ':' separated files at the beginning
|
||||
* of the fileName
|
||||
*/
|
||||
int lastColon = fileName.lastIndexOf(':');
|
||||
if (lastColon > 0) {
|
||||
if (lastColon+1 >= fileName.length()) {
|
||||
continue;
|
||||
}
|
||||
fileName = fileName.substring(lastColon+1);
|
||||
}
|
||||
String fontPart = st.sval.substring(breakPos+1);
|
||||
String fontID = specificFontIDForName(fontPart);
|
||||
String sVal = (String) fontNameMap.get(fontID);
|
||||
|
||||
if (FontUtilities.debugFonts()) {
|
||||
Logger logger = FontUtilities.getLogger();
|
||||
logger.info("file=" + fileName +
|
||||
" xlfd=" + fontPart);
|
||||
logger.info("fontID=" + fontID +
|
||||
" sVal=" + sVal);
|
||||
}
|
||||
String fullPath = null;
|
||||
try {
|
||||
File file = new File(path,fileName);
|
||||
/* we may have a resolved symbolic link
|
||||
* this becomes important for an xlfd we
|
||||
* still need to know the location it was
|
||||
* found to update the X server font path
|
||||
* for use by AWT heavyweights - and when 2D
|
||||
* wants to use the native rasteriser.
|
||||
*/
|
||||
if (xFontDirsMap == null) {
|
||||
xFontDirsMap = new HashMap();
|
||||
}
|
||||
xFontDirsMap.put(fontID, path);
|
||||
fullPath = file.getCanonicalPath();
|
||||
} catch (IOException e) {
|
||||
fullPath = path + File.separator + fileName;
|
||||
}
|
||||
Vector xVal = (Vector) xlfdMap.get(fullPath);
|
||||
if (FontUtilities.debugFonts()) {
|
||||
FontUtilities.getLogger()
|
||||
.info("fullPath=" + fullPath +
|
||||
" xVal=" + xVal);
|
||||
}
|
||||
if ((xVal == null || !xVal.contains(fontPart)) &&
|
||||
(sVal == null) || !sVal.startsWith("/")) {
|
||||
if (FontUtilities.debugFonts()) {
|
||||
FontUtilities.getLogger()
|
||||
.info("Map fontID:"+fontID +
|
||||
"to file:" + fullPath);
|
||||
}
|
||||
fontNameMap.put(fontID, fullPath);
|
||||
if (xVal == null) {
|
||||
xVal = new Vector();
|
||||
xlfdMap.put (fullPath, xVal);
|
||||
}
|
||||
xVal.add(fontPart);
|
||||
}
|
||||
|
||||
ttype = st.nextToken();
|
||||
if (ttype != StreamTokenizer.TT_EOL) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
fr.close();
|
||||
}
|
||||
} catch (IOException ioe1) {
|
||||
} finally {
|
||||
if (fr != null) {
|
||||
try {
|
||||
fr.close();
|
||||
} catch (IOException ioe2) {
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
public void loadFonts() {
|
||||
super.loadFonts();
|
||||
/* These maps are greatly expanded during a loadFonts but
|
||||
* can be reset to their initial state afterwards.
|
||||
* Since preferLocaleFonts() and preferProportionalFonts() will
|
||||
* trigger a partial repopulating from the FontConfiguration
|
||||
* it has to be the inital (empty) state for the latter two, not
|
||||
* simply nulling out.
|
||||
* xFontDirsMap is a special case in that the implementation
|
||||
* will typically not ever need to initialise it so it can be null.
|
||||
*/
|
||||
xFontDirsMap = null;
|
||||
xlfdMap = new HashMap(1);
|
||||
fontNameMap = new HashMap(1);
|
||||
}
|
||||
|
||||
private String getObliqueLucidaFontID(String fontID) {
|
||||
if (fontID.startsWith("-lucidasans-medium-i-normal") ||
|
||||
fontID.startsWith("-lucidasans-bold-i-normal") ||
|
||||
fontID.startsWith("-lucidatypewriter-medium-i-normal") ||
|
||||
fontID.startsWith("-lucidatypewriter-bold-i-normal")) {
|
||||
return fontID.substring(0, fontID.indexOf("-i-"));
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
private static String getX11FontName(String platName) {
|
||||
String xlfd = platName.replaceAll("%d", "*");
|
||||
if (NativeFont.fontExists(xlfd)) {
|
||||
return xlfd;
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
private void initObliqueLucidaFontMap() {
|
||||
oblmap = new HashMap<String, String>();
|
||||
oblmap.put("-lucidasans-medium",
|
||||
jreLibDirName+"/fonts/LucidaSansRegular.ttf");
|
||||
oblmap.put("-lucidasans-bold",
|
||||
jreLibDirName+"/fonts/LucidaSansDemiBold.ttf");
|
||||
oblmap.put("-lucidatypewriter-medium",
|
||||
jreLibDirName+"/fonts/LucidaTypewriterRegular.ttf");
|
||||
oblmap.put("-lucidatypewriter-bold",
|
||||
jreLibDirName+"/fonts/LucidaTypewriterBold.ttf");
|
||||
}
|
||||
|
||||
private boolean isHeadless() {
|
||||
GraphicsEnvironment ge =
|
||||
GraphicsEnvironment.getLocalGraphicsEnvironment();
|
||||
return GraphicsEnvironment.isHeadless();
|
||||
}
|
||||
|
||||
private String specificFontIDForName(String name) {
|
||||
|
||||
int[] hPos = new int[14];
|
||||
int hyphenCnt = 1;
|
||||
int pos = 1;
|
||||
|
||||
while (pos != -1 && hyphenCnt < 14) {
|
||||
pos = name.indexOf('-', pos);
|
||||
if (pos != -1) {
|
||||
hPos[hyphenCnt++] = pos;
|
||||
pos++;
|
||||
}
|
||||
}
|
||||
|
||||
if (hyphenCnt != 14) {
|
||||
if (FontUtilities.debugFonts()) {
|
||||
FontUtilities.getLogger()
|
||||
.severe("Font Configuration Font ID is malformed:" + name);
|
||||
}
|
||||
return name; // what else can we do?
|
||||
}
|
||||
|
||||
StringBuffer sb =
|
||||
new StringBuffer(name.substring(hPos[FAMILY_NAME_FIELD-1],
|
||||
hPos[SETWIDTH_NAME_FIELD]));
|
||||
sb.append(name.substring(hPos[CHARSET_REGISTRY_FIELD-1]));
|
||||
String retval = sb.toString().toLowerCase (Locale.ENGLISH);
|
||||
return retval;
|
||||
}
|
||||
|
||||
private String switchFontIDForName(String name) {
|
||||
|
||||
int[] hPos = new int[14];
|
||||
int hyphenCnt = 1;
|
||||
int pos = 1;
|
||||
|
||||
while (pos != -1 && hyphenCnt < 14) {
|
||||
pos = name.indexOf('-', pos);
|
||||
if (pos != -1) {
|
||||
hPos[hyphenCnt++] = pos;
|
||||
pos++;
|
||||
}
|
||||
}
|
||||
|
||||
if (hyphenCnt != 14) {
|
||||
if (FontUtilities.debugFonts()) {
|
||||
FontUtilities.getLogger()
|
||||
.severe("Font Configuration Font ID is malformed:" + name);
|
||||
}
|
||||
return name; // what else can we do?
|
||||
}
|
||||
|
||||
String slant = name.substring(hPos[SLANT_FIELD-1]+1,
|
||||
hPos[SLANT_FIELD]);
|
||||
String family = name.substring(hPos[FAMILY_NAME_FIELD-1]+1,
|
||||
hPos[FAMILY_NAME_FIELD]);
|
||||
String registry = name.substring(hPos[CHARSET_REGISTRY_FIELD-1]+1,
|
||||
hPos[CHARSET_REGISTRY_FIELD]);
|
||||
String encoding = name.substring(hPos[CHARSET_ENCODING_FIELD-1]+1);
|
||||
|
||||
if (slant.equals("i")) {
|
||||
slant = "o";
|
||||
} else if (slant.equals("o")) {
|
||||
slant = "i";
|
||||
}
|
||||
// workaround for #4471000
|
||||
if (family.equals("itc zapfdingbats")
|
||||
&& registry.equals("sun")
|
||||
&& encoding.equals("fontspecific")){
|
||||
registry = "adobe";
|
||||
}
|
||||
StringBuffer sb =
|
||||
new StringBuffer(name.substring(hPos[FAMILY_NAME_FIELD-1],
|
||||
hPos[SLANT_FIELD-1]+1));
|
||||
sb.append(slant);
|
||||
sb.append(name.substring(hPos[SLANT_FIELD],
|
||||
hPos[SETWIDTH_NAME_FIELD]+1));
|
||||
sb.append(registry);
|
||||
sb.append(name.substring(hPos[CHARSET_ENCODING_FIELD-1]));
|
||||
String retval = sb.toString().toLowerCase (Locale.ENGLISH);
|
||||
return retval;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns the face name for the given XLFD.
|
||||
*/
|
||||
public String getFileNameFromXLFD(String name) {
|
||||
String fileName = null;
|
||||
String fontID = specificFontIDForName(name);
|
||||
if (fontID != null) {
|
||||
fileName = (String)fontNameMap.get(fontID);
|
||||
if (fileName == null) {
|
||||
fontID = switchFontIDForName(name);
|
||||
fileName = (String)fontNameMap.get(fontID);
|
||||
}
|
||||
if (fileName == null) {
|
||||
fileName = getDefaultFontFile();
|
||||
}
|
||||
}
|
||||
return fileName;
|
||||
}
|
||||
|
||||
/* Register just the paths, (it doesn't register the fonts).
|
||||
* If a font configuration file has specified a baseFontPath
|
||||
* fontPath is just those directories, unless on usage we
|
||||
* find it doesn't contain what we need for the logical fonts.
|
||||
* Otherwise, we register all the paths on Solaris, because
|
||||
* the fontPath we have here is the complete one from
|
||||
* parsing /var/sadm/install/contents, not just
|
||||
* what's on the X font path (may be this should be
|
||||
* changed).
|
||||
* But for now what it means is that if we didn't do
|
||||
* this then if the font weren't listed anywhere on the
|
||||
* less complete font path we'd trigger loadFonts which
|
||||
* actually registers the fonts. This may actually be
|
||||
* the right thing tho' since that would also set up
|
||||
* the X font path without which we wouldn't be able to
|
||||
* display some "native" fonts.
|
||||
* So something to revisit is that probably fontPath
|
||||
* here ought to be only the X font path + jre font dir.
|
||||
* loadFonts should have a separate native call to
|
||||
* get the rest of the platform font path.
|
||||
*
|
||||
* Registering the directories can now be avoided in the
|
||||
* font configuration initialisation when filename entries
|
||||
* exist in the font configuration file for all fonts.
|
||||
* (Perhaps a little confusingly a filename entry is
|
||||
* actually keyed using the XLFD used in the font entries,
|
||||
* and it maps *to* a real filename).
|
||||
* In the event any are missing, registration of all
|
||||
* directories will be invoked to find the real files.
|
||||
*
|
||||
* But registering the directory performed other
|
||||
* functions such as filling in the map of all native names
|
||||
* for the font. So when this method isn't invoked, they still
|
||||
* must be found. This is mitigated by getNativeNames now
|
||||
* being able to return at least the platform name, but mostly
|
||||
* by ensuring that when a filename key is found, that
|
||||
* xlfd key is stored as one of the set of platform names
|
||||
* for the font. Its a set because typical font configuration
|
||||
* files reference the same CJK font files using multiple
|
||||
* X11 encodings. For the code that adds this to the map
|
||||
* see X11GE.getFileNameFromPlatformName(..)
|
||||
* If you don't get all of these then some code points may
|
||||
* not use the Xserver, and will not get the PCF bitmaps
|
||||
* that are available for some point sizes.
|
||||
* So, in the event that there is such a problem,
|
||||
* unconditionally making this call may be necessary, at
|
||||
* some cost to JRE start-up
|
||||
*/
|
||||
@Override
|
||||
protected void registerFontDirs(String pathName) {
|
||||
|
||||
StringTokenizer parser = new StringTokenizer(pathName,
|
||||
File.pathSeparator);
|
||||
try {
|
||||
while (parser.hasMoreTokens()) {
|
||||
String dirPath = parser.nextToken();
|
||||
if (dirPath != null && !registeredDirs.containsKey(dirPath)) {
|
||||
registeredDirs.put(dirPath, null);
|
||||
registerFontDir(dirPath);
|
||||
}
|
||||
}
|
||||
} catch (NoSuchElementException e) {
|
||||
}
|
||||
}
|
||||
|
||||
// An X font spec (xlfd) includes an encoding. The same TrueType font file
|
||||
// may be referenced from different X font directories in font.dir files
|
||||
// to support use in multiple encodings by X apps.
|
||||
// So for the purposes of font configuration logical fonts where AWT
|
||||
// heavyweights need to access the font via X APIs we need to ensure that
|
||||
// the directory for precisely the encodings needed by this are added to
|
||||
// the x font path. This requires that we note the platform names
|
||||
// specified in font configuration files and use that to identify the
|
||||
// X font directory that contains a font.dir file for that platform name
|
||||
// and add it to the X font path (if display is local)
|
||||
// Here we make use of an already built map of xlfds to font locations
|
||||
// to add the font location to the set of those required to build the
|
||||
// x font path needed by AWT.
|
||||
// These are added to the x font path later.
|
||||
// All this is necessary because on Solaris the font.dir directories
|
||||
// may contain not real font files, but symbolic links to the actual
|
||||
// location but that location is not suitable for the x font path, since
|
||||
// it probably doesn't have a font.dir at all and certainly not one
|
||||
// with the required encodings
|
||||
// If the fontconfiguration file is properly set up so that all fonts
|
||||
// are mapped to files then we will never trigger initialising
|
||||
// xFontDirsMap (it will be null). In this case the awtfontpath entries
|
||||
// must specify all the X11 directories needed by AWT.
|
||||
@Override
|
||||
protected void addFontToPlatformFontPath(String platformName) {
|
||||
// Lazily initialize fontConfigDirs.
|
||||
getPlatformFontPathFromFontConfig();
|
||||
if (xFontDirsMap != null) {
|
||||
String fontID = specificFontIDForName(platformName);
|
||||
String dirName = (String)xFontDirsMap.get(fontID);
|
||||
if (dirName != null) {
|
||||
fontConfigDirs.add(dirName);
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
private void getPlatformFontPathFromFontConfig() {
|
||||
if (fontConfigDirs == null) {
|
||||
fontConfigDirs = getFontConfiguration().getAWTFontPathSet();
|
||||
if (FontUtilities.debugFonts() && fontConfigDirs != null) {
|
||||
String[] names = fontConfigDirs.toArray(new String[0]);
|
||||
for (int i=0;i<names.length;i++) {
|
||||
FontUtilities.getLogger().info("awtfontpath : " + names[i]);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void registerPlatformFontsUsedByFontConfiguration() {
|
||||
// Lazily initialize fontConfigDirs.
|
||||
getPlatformFontPathFromFontConfig();
|
||||
if (fontConfigDirs == null) {
|
||||
return;
|
||||
}
|
||||
if (FontUtilities.isLinux) {
|
||||
fontConfigDirs.add(jreLibDirName+File.separator+"oblique-fonts");
|
||||
}
|
||||
fontdirs = (String[])fontConfigDirs.toArray(new String[0]);
|
||||
}
|
||||
|
||||
/* Called by MToolkit to set the X11 font path */
|
||||
public static void setNativeFontPath() {
|
||||
if (fontdirs == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
// need to register these individually rather than by one call
|
||||
// to ensure that one bad directory doesn't cause all to be rejected
|
||||
for (int i=0; i<fontdirs.length; i++) {
|
||||
if (FontUtilities.debugFonts()) {
|
||||
FontUtilities.getLogger().info("Add " + fontdirs[i] + " to X11 fontpath");
|
||||
}
|
||||
setNativeFontPath(fontdirs[i]);
|
||||
}
|
||||
}
|
||||
|
||||
private synchronized static native void setNativeFontPath(String fontPath);
|
||||
|
||||
|
||||
// Implements SunGraphicsEnvironment.createFontConfiguration.
|
||||
protected FontConfiguration createFontConfiguration() {
|
||||
/* The logic here decides whether to use a preconfigured
|
||||
* fontconfig.properties file, or synthesise one using platform APIs.
|
||||
* On Solaris (as opposed to OpenSolaris) we try to use the
|
||||
* pre-configured ones, but if the files it specifies are missing
|
||||
* we fail-safe to synthesising one. This might happen if Solaris
|
||||
* changes its fonts.
|
||||
* For OpenSolaris I don't expect us to ever create fontconfig files,
|
||||
* so it will always synthesise. Note that if we misidentify
|
||||
* OpenSolaris as Solaris, then the test for the presence of
|
||||
* Solaris-only font files will correct this.
|
||||
* For Linux we require an exact match of distro and version to
|
||||
* use the preconfigured file, and also that it points to
|
||||
* existent fonts.
|
||||
* If synthesising fails, we fall back to any preconfigured file
|
||||
* and do the best we can. For the commercial JDK this will be
|
||||
* fine as it includes the Lucida fonts. OpenJDK should not hit
|
||||
* this as the synthesis should always work on its platforms.
|
||||
*/
|
||||
FontConfiguration mFontConfig = new MFontConfiguration(this);
|
||||
if (FontUtilities.isOpenSolaris ||
|
||||
(FontUtilities.isLinux &&
|
||||
(!mFontConfig.foundOsSpecificFile() ||
|
||||
!mFontConfig.fontFilesArePresent()) ||
|
||||
(FontUtilities.isSolaris && !mFontConfig.fontFilesArePresent()))) {
|
||||
FcFontConfiguration fcFontConfig =
|
||||
new FcFontConfiguration(this);
|
||||
if (fcFontConfig.init()) {
|
||||
return fcFontConfig;
|
||||
}
|
||||
}
|
||||
mFontConfig.init();
|
||||
return mFontConfig;
|
||||
}
|
||||
public FontConfiguration
|
||||
createFontConfiguration(boolean preferLocaleFonts,
|
||||
boolean preferPropFonts) {
|
||||
|
||||
return new MFontConfiguration(this,
|
||||
preferLocaleFonts, preferPropFonts);
|
||||
}
|
||||
|
||||
public synchronized native String getFontPath(boolean noType1Fonts);
|
||||
|
||||
public String[] getDefaultPlatformFont() {
|
||||
if (defaultPlatformFont != null) {
|
||||
return defaultPlatformFont;
|
||||
}
|
||||
String[] info = new String[2];
|
||||
getFontConfigManager().initFontConfigFonts(false);
|
||||
FontConfigManager.FcCompFont[] fontConfigFonts =
|
||||
getFontConfigManager().getFontConfigFonts();
|
||||
for (int i=0; i<fontConfigFonts.length; i++) {
|
||||
if ("sans".equals(fontConfigFonts[i].fcFamily) &&
|
||||
0 == fontConfigFonts[i].style) {
|
||||
info[0] = fontConfigFonts[i].firstFont.familyName;
|
||||
info[1] = fontConfigFonts[i].firstFont.fontFile;
|
||||
break;
|
||||
}
|
||||
}
|
||||
/* Absolute last ditch attempt in the face of fontconfig problems.
|
||||
* If we didn't match, pick the first, or just make something
|
||||
* up so we don't NPE.
|
||||
*/
|
||||
if (info[0] == null) {
|
||||
if (fontConfigFonts.length > 0 &&
|
||||
fontConfigFonts[0].firstFont.fontFile != null) {
|
||||
info[0] = fontConfigFonts[0].firstFont.familyName;
|
||||
info[1] = fontConfigFonts[0].firstFont.fontFile;
|
||||
} else {
|
||||
info[0] = "Dialog";
|
||||
info[1] = "/dialog.ttf";
|
||||
}
|
||||
}
|
||||
defaultPlatformFont = info;
|
||||
return defaultPlatformFont;
|
||||
}
|
||||
|
||||
public synchronized FontConfigManager getFontConfigManager() {
|
||||
|
||||
if (fcManager == null) {
|
||||
fcManager = new FontConfigManager();
|
||||
}
|
||||
|
||||
return fcManager;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected FontUIResource getFontConfigFUIR(String family, int style, int size) {
|
||||
|
||||
CompositeFont font2D = getFontConfigManager().getFontConfigFont(family, style);
|
||||
|
||||
if (font2D == null) { // Not expected, just a precaution.
|
||||
return new FontUIResource(family, style, size);
|
||||
}
|
||||
|
||||
/* The name of the font will be that of the physical font in slot,
|
||||
* but by setting the handle to that of the CompositeFont it
|
||||
* renders as that CompositeFont.
|
||||
* It also needs to be marked as a created font which is the
|
||||
* current mechanism to signal that deriveFont etc must copy
|
||||
* the handle from the original font.
|
||||
*/
|
||||
FontUIResource fuir =
|
||||
new FontUIResource(font2D.getFamilyName(null), style, size);
|
||||
FontAccess.getFontAccess().setFont2D(fuir, font2D.handle);
|
||||
FontAccess.getFontAccess().setCreatedFont(fuir);
|
||||
return fuir;
|
||||
}
|
||||
}
|
454
jdk/src/solaris/classes/sun/font/FontConfigManager.java
Normal file
454
jdk/src/solaris/classes/sun/font/FontConfigManager.java
Normal file
@ -0,0 +1,454 @@
|
||||
/*
|
||||
* Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
|
||||
* particular file as subject to the "Classpath" exception as provided
|
||||
* by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||||
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||||
* have any questions.
|
||||
*/
|
||||
|
||||
package sun.font;
|
||||
|
||||
import java.util.Locale;
|
||||
import java.util.logging.Logger;
|
||||
|
||||
import sun.awt.SunHints;
|
||||
import sun.awt.SunToolkit;
|
||||
|
||||
/**
|
||||
* Small utility class to manage FontConfig.
|
||||
*/
|
||||
public class FontConfigManager {
|
||||
|
||||
static boolean fontConfigFailed = false;
|
||||
|
||||
/* This is populated by native */
|
||||
private static final FontConfigInfo fcInfo = new FontConfigInfo();
|
||||
|
||||
/* Begin support for GTK Look and Feel - query libfontconfig and
|
||||
* return a composite Font to Swing that uses the desktop font(s).
|
||||
*/
|
||||
|
||||
/* These next three classes are just data structures.
|
||||
*/
|
||||
public static class FontConfigFont {
|
||||
public String familyName; // eg Bitstream Vera Sans
|
||||
public String styleStr; // eg Bold
|
||||
public String fullName; // eg Bitstream Vera Sans Bold
|
||||
public String fontFile; // eg /usr/X11/lib/fonts/foo.ttf
|
||||
}
|
||||
|
||||
public static class FcCompFont {
|
||||
public String fcName; // eg sans
|
||||
public String fcFamily; // eg sans
|
||||
public String jdkName; // eg sansserif
|
||||
public int style; // eg 0=PLAIN
|
||||
public FontConfigFont firstFont;
|
||||
public FontConfigFont[] allFonts;
|
||||
//boolean preferBitmaps; // if embedded bitmaps preferred over AA
|
||||
public CompositeFont compFont; // null if not yet created/known.
|
||||
}
|
||||
|
||||
public static class FontConfigInfo {
|
||||
public int fcVersion;
|
||||
public String[] cacheDirs = new String[4];
|
||||
}
|
||||
|
||||
/* fontconfig recognises slants roman, italic, as well as oblique,
|
||||
* and a slew of weights, where the ones that matter here are
|
||||
* regular and bold.
|
||||
* To fully qualify what we want, we can for example ask for (eg)
|
||||
* Font.PLAIN : "serif:regular:roman"
|
||||
* Font.BOLD : "serif:bold:roman"
|
||||
* Font.ITALIC : "serif:regular:italic"
|
||||
* Font.BOLD|Font.ITALIC : "serif:bold:italic"
|
||||
*/
|
||||
private static String[] fontConfigNames = {
|
||||
"sans:regular:roman",
|
||||
"sans:bold:roman",
|
||||
"sans:regular:italic",
|
||||
"sans:bold:italic",
|
||||
|
||||
"serif:regular:roman",
|
||||
"serif:bold:roman",
|
||||
"serif:regular:italic",
|
||||
"serif:bold:italic",
|
||||
|
||||
"monospace:regular:roman",
|
||||
"monospace:bold:roman",
|
||||
"monospace:regular:italic",
|
||||
"monospace:bold:italic",
|
||||
};
|
||||
|
||||
/* This array has the array elements created in Java code and is
|
||||
* passed down to native to be filled in.
|
||||
*/
|
||||
private FcCompFont[] fontConfigFonts;
|
||||
|
||||
/**
|
||||
* Instantiates a new FontConfigManager getting the default instance
|
||||
* of FontManager from the FontManagerFactory.
|
||||
*/
|
||||
public FontConfigManager() {
|
||||
}
|
||||
|
||||
public static String[] getFontConfigNames() {
|
||||
return fontConfigNames;
|
||||
}
|
||||
|
||||
/* Called from code that needs to know what are the AA settings
|
||||
* that apps using FC would pick up for the default desktop font.
|
||||
* Note apps can change the default desktop font. etc, so this
|
||||
* isn't certain to be right but its going to correct for most cases.
|
||||
* Native return values map to the text aa values in sun.awt.SunHints.
|
||||
* which is used to look up the renderinghint value object.
|
||||
*/
|
||||
public static Object getFontConfigAAHint() {
|
||||
return getFontConfigAAHint("sans");
|
||||
}
|
||||
|
||||
/* This is public solely so that for debugging purposes it can be called
|
||||
* with other names, which might (eg) include a size, eg "sans-24"
|
||||
* The return value is a text aa rendering hint value.
|
||||
* Normally we should call the no-args version.
|
||||
*/
|
||||
public static Object getFontConfigAAHint(String fcFamily) {
|
||||
if (FontUtilities.isWindows) {
|
||||
return null;
|
||||
} else {
|
||||
int hint = getFontConfigAASettings(getFCLocaleStr(), fcFamily);
|
||||
if (hint < 0) {
|
||||
return null;
|
||||
} else {
|
||||
return SunHints.Value.get(SunHints.INTKEY_TEXT_ANTIALIASING,
|
||||
hint);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private static String getFCLocaleStr() {
|
||||
Locale l = SunToolkit.getStartupLocale();
|
||||
String localeStr = l.getLanguage();
|
||||
String country = l.getCountry();
|
||||
if (!country.equals("")) {
|
||||
localeStr = localeStr + "-" + country;
|
||||
}
|
||||
return localeStr;
|
||||
}
|
||||
|
||||
/* This does cause the native libfontconfig to be loaded and unloaded,
|
||||
* but it does not incur the overhead of initialisation of its
|
||||
* data structures, so shouldn't have a measurable impact.
|
||||
*/
|
||||
public static native int getFontConfigVersion();
|
||||
|
||||
/* This can be made public if it's needed to force a re-read
|
||||
* rather than using the cached values. The re-read would be needed
|
||||
* only if some event signalled that the fontconfig has changed.
|
||||
* In that event this method would need to return directly the array
|
||||
* to be used by the caller in case it subsequently changed.
|
||||
*/
|
||||
public synchronized void initFontConfigFonts(boolean includeFallbacks) {
|
||||
|
||||
if (fontConfigFonts != null) {
|
||||
if (!includeFallbacks || (fontConfigFonts[0].allFonts != null)) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
if (FontUtilities.isWindows || fontConfigFailed) {
|
||||
return;
|
||||
}
|
||||
|
||||
long t0 = 0;
|
||||
if (FontUtilities.isLogging()) {
|
||||
t0 = System.nanoTime();
|
||||
}
|
||||
|
||||
String[] fontConfigNames = FontConfigManager.getFontConfigNames();
|
||||
FcCompFont[] fontArr = new FcCompFont[fontConfigNames.length];
|
||||
|
||||
for (int i = 0; i< fontArr.length; i++) {
|
||||
fontArr[i] = new FcCompFont();
|
||||
fontArr[i].fcName = fontConfigNames[i];
|
||||
int colonPos = fontArr[i].fcName.indexOf(':');
|
||||
fontArr[i].fcFamily = fontArr[i].fcName.substring(0, colonPos);
|
||||
fontArr[i].jdkName = FontUtilities.mapFcName(fontArr[i].fcFamily);
|
||||
fontArr[i].style = i % 4; // depends on array order.
|
||||
}
|
||||
getFontConfig(getFCLocaleStr(), fcInfo, fontArr, includeFallbacks);
|
||||
/* If don't find anything (eg no libfontconfig), then just return */
|
||||
for (int i = 0; i< fontArr.length; i++) {
|
||||
FcCompFont fci = fontArr[i];
|
||||
if (fci.firstFont == null) {
|
||||
if (FontUtilities.isLogging()) {
|
||||
Logger logger = FontUtilities.getLogger();
|
||||
logger.info("Fontconfig returned no fonts.");
|
||||
}
|
||||
fontConfigFailed = true;
|
||||
return;
|
||||
}
|
||||
}
|
||||
fontConfigFonts = fontArr;
|
||||
|
||||
if (FontUtilities.isLogging()) {
|
||||
|
||||
Logger logger = FontUtilities.getLogger();
|
||||
|
||||
long t1 = System.nanoTime();
|
||||
logger.info("Time spent accessing fontconfig="
|
||||
+ ((t1 - t0) / 1000000) + "ms.");
|
||||
|
||||
for (int i = 0; i< fontConfigFonts.length; i++) {
|
||||
FcCompFont fci = fontConfigFonts[i];
|
||||
logger.info("FC font " + fci.fcName+" maps to family " +
|
||||
fci.firstFont.familyName +
|
||||
" in file " + fci.firstFont.fontFile);
|
||||
if (fci.allFonts != null) {
|
||||
for (int f=0;f<fci.allFonts.length;f++) {
|
||||
FontConfigFont fcf = fci.allFonts[f];
|
||||
logger.info("Family=" + fcf.familyName +
|
||||
" Style="+ fcf.styleStr +
|
||||
" Fullname="+fcf.fullName +
|
||||
" File="+fcf.fontFile);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public PhysicalFont registerFromFcInfo(FcCompFont fcInfo) {
|
||||
|
||||
SunFontManager fm = SunFontManager.getInstance();
|
||||
|
||||
/* If it's a TTC file we need to know that as we will need to
|
||||
* make sure we return the right font */
|
||||
String fontFile = fcInfo.firstFont.fontFile;
|
||||
int offset = fontFile.length()-4;
|
||||
if (offset <= 0) {
|
||||
return null;
|
||||
}
|
||||
String ext = fontFile.substring(offset).toLowerCase();
|
||||
boolean isTTC = ext.equals(".ttc");
|
||||
|
||||
/* If this file is already registered, can just return its font.
|
||||
* However we do need to check in case it's a TTC as we need
|
||||
* a specific font, so rather than directly returning it, let
|
||||
* findFont2D resolve that.
|
||||
*/
|
||||
PhysicalFont physFont = fm.getRegisteredFontFile(fontFile);
|
||||
if (physFont != null) {
|
||||
if (isTTC) {
|
||||
Font2D f2d = fm.findFont2D(fcInfo.firstFont.familyName,
|
||||
fcInfo.style,
|
||||
FontManager.NO_FALLBACK);
|
||||
if (f2d instanceof PhysicalFont) { /* paranoia */
|
||||
return (PhysicalFont)f2d;
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
} else {
|
||||
return physFont;
|
||||
}
|
||||
}
|
||||
|
||||
/* If the font may hide a JRE font (eg fontconfig says it is
|
||||
* Lucida Sans), we want to use the JRE version, so make it
|
||||
* point to the JRE font.
|
||||
*/
|
||||
physFont = fm.findJREDeferredFont(fcInfo.firstFont.familyName,
|
||||
fcInfo.style);
|
||||
|
||||
/* It is also possible the font file is on the "deferred" list,
|
||||
* in which case we can just initialise it now.
|
||||
*/
|
||||
if (physFont == null &&
|
||||
fm.isDeferredFont(fontFile) == true) {
|
||||
physFont = fm.initialiseDeferredFont(fcInfo.firstFont.fontFile);
|
||||
/* use findFont2D to get the right font from TTC's */
|
||||
if (physFont != null) {
|
||||
if (isTTC) {
|
||||
Font2D f2d = fm.findFont2D(fcInfo.firstFont.familyName,
|
||||
fcInfo.style,
|
||||
FontManager.NO_FALLBACK);
|
||||
if (f2d instanceof PhysicalFont) { /* paranoia */
|
||||
return (PhysicalFont)f2d;
|
||||
} else {
|
||||
return null;
|
||||
}
|
||||
} else {
|
||||
return physFont;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* In the majority of cases we reach here, and need to determine
|
||||
* the type and rank to register the font.
|
||||
*/
|
||||
if (physFont == null) {
|
||||
int fontFormat = SunFontManager.FONTFORMAT_NONE;
|
||||
int fontRank = Font2D.UNKNOWN_RANK;
|
||||
|
||||
if (ext.equals(".ttf") || isTTC) {
|
||||
fontFormat = SunFontManager.FONTFORMAT_TRUETYPE;
|
||||
fontRank = Font2D.TTF_RANK;
|
||||
} else if (ext.equals(".pfa") || ext.equals(".pfb")) {
|
||||
fontFormat = SunFontManager.FONTFORMAT_TYPE1;
|
||||
fontRank = Font2D.TYPE1_RANK;
|
||||
}
|
||||
physFont = fm.registerFontFile(fcInfo.firstFont.fontFile, null,
|
||||
fontFormat, true, fontRank);
|
||||
}
|
||||
return physFont;
|
||||
}
|
||||
|
||||
/*
|
||||
* We need to return a Composite font which has as the font in
|
||||
* its first slot one obtained from fontconfig.
|
||||
*/
|
||||
public CompositeFont getFontConfigFont(String name, int style) {
|
||||
|
||||
name = name.toLowerCase();
|
||||
|
||||
initFontConfigFonts(false);
|
||||
|
||||
FcCompFont fcInfo = null;
|
||||
for (int i=0; i<fontConfigFonts.length; i++) {
|
||||
if (name.equals(fontConfigFonts[i].fcFamily) &&
|
||||
style == fontConfigFonts[i].style) {
|
||||
fcInfo = fontConfigFonts[i];
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (fcInfo == null) {
|
||||
fcInfo = fontConfigFonts[0];
|
||||
}
|
||||
|
||||
if (FontUtilities.isLogging()) {
|
||||
FontUtilities.getLogger()
|
||||
.info("FC name=" + name + " style=" + style +
|
||||
" uses " + fcInfo.firstFont.familyName +
|
||||
" in file: " + fcInfo.firstFont.fontFile);
|
||||
}
|
||||
|
||||
if (fcInfo.compFont != null) {
|
||||
return fcInfo.compFont;
|
||||
}
|
||||
|
||||
/* jdkFont is going to be used for slots 1..N and as a fallback.
|
||||
* Slot 0 will be the physical font from fontconfig.
|
||||
*/
|
||||
FontManager fm = FontManagerFactory.getInstance();
|
||||
CompositeFont jdkFont = (CompositeFont)
|
||||
fm.findFont2D(fcInfo.jdkName, style, FontManager.LOGICAL_FALLBACK);
|
||||
|
||||
if (fcInfo.firstFont.familyName == null ||
|
||||
fcInfo.firstFont.fontFile == null) {
|
||||
return (fcInfo.compFont = jdkFont);
|
||||
}
|
||||
|
||||
/* First, see if the family and exact style is already registered.
|
||||
* If it is, use it. If it's not, then try to register it.
|
||||
* If that registration fails (signalled by null) just return the
|
||||
* regular JDK composite.
|
||||
* Algorithmically styled fonts won't match on exact style, so
|
||||
* will fall through this code, but the regisration code will
|
||||
* find that file already registered and return its font.
|
||||
*/
|
||||
FontFamily family = FontFamily.getFamily(fcInfo.firstFont.familyName);
|
||||
PhysicalFont physFont = null;
|
||||
if (family != null) {
|
||||
Font2D f2D = family.getFontWithExactStyleMatch(fcInfo.style);
|
||||
if (f2D instanceof PhysicalFont) {
|
||||
physFont = (PhysicalFont)f2D;
|
||||
}
|
||||
}
|
||||
|
||||
if (physFont == null ||
|
||||
!fcInfo.firstFont.fontFile.equals(physFont.platName)) {
|
||||
physFont = registerFromFcInfo(fcInfo);
|
||||
if (physFont == null) {
|
||||
return (fcInfo.compFont = jdkFont);
|
||||
}
|
||||
family = FontFamily.getFamily(physFont.getFamilyName(null));
|
||||
}
|
||||
|
||||
/* Now register the fonts in the family (the other styles) after
|
||||
* checking that they aren't already registered and are actually in
|
||||
* a different file. They may be the same file in CJK cases.
|
||||
* For cases where they are different font files - eg as is common for
|
||||
* Latin fonts, then we rely on fontconfig to report these correctly.
|
||||
* Assume that all styles of this font are found by fontconfig,
|
||||
* so we can find all the family members which must be registered
|
||||
* together to prevent synthetic styling.
|
||||
*/
|
||||
for (int i=0; i<fontConfigFonts.length; i++) {
|
||||
FcCompFont fc = fontConfigFonts[i];
|
||||
if (fc != fcInfo &&
|
||||
physFont.getFamilyName(null).equals(fc.firstFont.familyName) &&
|
||||
!fc.firstFont.fontFile.equals(physFont.platName) &&
|
||||
family.getFontWithExactStyleMatch(fc.style) == null) {
|
||||
|
||||
registerFromFcInfo(fontConfigFonts[i]);
|
||||
}
|
||||
}
|
||||
|
||||
/* Now we have a physical font. We will back this up with the JDK
|
||||
* logical font (sansserif, serif, or monospaced) that corresponds
|
||||
* to the Pango/GTK/FC logical font name.
|
||||
*/
|
||||
return (fcInfo.compFont = new CompositeFont(physFont, jdkFont));
|
||||
}
|
||||
|
||||
/**
|
||||
*
|
||||
* @param locale
|
||||
* @param fcFamily
|
||||
* @return
|
||||
*/
|
||||
public FcCompFont[] getFontConfigFonts() {
|
||||
return fontConfigFonts;
|
||||
}
|
||||
|
||||
/* Return an array of FcCompFont structs describing the primary
|
||||
* font located for each of fontconfig/GTK/Pango's logical font names.
|
||||
*/
|
||||
private static native void getFontConfig(String locale,
|
||||
FontConfigInfo fcInfo,
|
||||
FcCompFont[] fonts,
|
||||
boolean includeFallbacks);
|
||||
|
||||
void populateFontConfig(FcCompFont[] fcInfo) {
|
||||
fontConfigFonts = fcInfo;
|
||||
}
|
||||
|
||||
FcCompFont[] loadFontConfig() {
|
||||
initFontConfigFonts(true);
|
||||
return fontConfigFonts;
|
||||
}
|
||||
|
||||
FontConfigInfo getFontConfigInfo() {
|
||||
initFontConfigFonts(true);
|
||||
return fcInfo;
|
||||
}
|
||||
|
||||
private static native int
|
||||
getFontConfigAASettings(String locale, String fcFamily);
|
||||
}
|
280
jdk/src/windows/classes/sun/awt/Win32FontManager.java
Normal file
280
jdk/src/windows/classes/sun/awt/Win32FontManager.java
Normal file
@ -0,0 +1,280 @@
|
||||
/*
|
||||
* Copyright 2008 Sun Microsystems, Inc. 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. Sun designates this
|
||||
* particular file as subject to the "Classpath" exception as provided
|
||||
* by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||||
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||||
* have any questions.
|
||||
*/
|
||||
|
||||
|
||||
package sun.awt;
|
||||
|
||||
import java.awt.FontFormatException;
|
||||
import java.awt.GraphicsEnvironment;
|
||||
import java.io.File;
|
||||
import java.security.AccessController;
|
||||
import java.security.PrivilegedAction;
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashMap;
|
||||
import java.util.Locale;
|
||||
import java.util.NoSuchElementException;
|
||||
import java.util.StringTokenizer;
|
||||
|
||||
import sun.awt.Win32GraphicsEnvironment;
|
||||
import sun.awt.windows.WFontConfiguration;
|
||||
import sun.font.FontManager;
|
||||
import sun.font.SunFontManager;
|
||||
import sun.font.TrueTypeFont;
|
||||
import sun.java2d.HeadlessGraphicsEnvironment;
|
||||
import sun.java2d.SunGraphicsEnvironment;
|
||||
|
||||
/**
|
||||
* The X11 implementation of {@link FontManager}.
|
||||
*/
|
||||
public class Win32FontManager extends SunFontManager {
|
||||
|
||||
private static String[] defaultPlatformFont = null;
|
||||
|
||||
private static TrueTypeFont eudcFont;
|
||||
|
||||
static {
|
||||
|
||||
AccessController.doPrivileged(new PrivilegedAction() {
|
||||
|
||||
public Object run() {
|
||||
String eudcFile = getEUDCFontFile();
|
||||
if (eudcFile != null) {
|
||||
try {
|
||||
eudcFont = new TrueTypeFont(eudcFile, null, 0,
|
||||
true);
|
||||
} catch (FontFormatException e) {
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
});
|
||||
}
|
||||
|
||||
/* Used on Windows to obtain from the windows registry the name
|
||||
* of a file containing the system EUFC font. If running in one of
|
||||
* the locales for which this applies, and one is defined, the font
|
||||
* defined by this file is appended to all composite fonts as a
|
||||
* fallback component.
|
||||
*/
|
||||
private static native String getEUDCFontFile();
|
||||
|
||||
public Win32FontManager() {
|
||||
super();
|
||||
AccessController.doPrivileged(new PrivilegedAction() {
|
||||
public Object run() {
|
||||
|
||||
/* Register the JRE fonts so that the native platform can
|
||||
* access them. This is used only on Windows so that when
|
||||
* printing the printer driver can access the fonts.
|
||||
*/
|
||||
registerJREFontsWithPlatform(jreFontDirName);
|
||||
return null;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
/* Unlike the shared code version, this expects a base file name -
|
||||
* not a full path name.
|
||||
* The font configuration file has base file names and the FontConfiguration
|
||||
* class reports these back to the GraphicsEnvironment, so these
|
||||
* are the componentFileNames of CompositeFonts.
|
||||
*/
|
||||
protected void registerFontFile(String fontFileName, String[] nativeNames,
|
||||
int fontRank, boolean defer) {
|
||||
|
||||
// REMIND: case compare depends on platform
|
||||
if (registeredFontFiles.contains(fontFileName)) {
|
||||
return;
|
||||
}
|
||||
registeredFontFiles.add(fontFileName);
|
||||
|
||||
int fontFormat;
|
||||
if (getTrueTypeFilter().accept(null, fontFileName)) {
|
||||
fontFormat = SunFontManager.FONTFORMAT_TRUETYPE;
|
||||
} else if (getType1Filter().accept(null, fontFileName)) {
|
||||
fontFormat = SunFontManager.FONTFORMAT_TYPE1;
|
||||
} else {
|
||||
/* on windows we don't use/register native fonts */
|
||||
return;
|
||||
}
|
||||
|
||||
if (fontPath == null) {
|
||||
fontPath = getPlatformFontPath(noType1Font);
|
||||
}
|
||||
|
||||
/* Look in the JRE font directory first.
|
||||
* This is playing it safe as we would want to find fonts in the
|
||||
* JRE font directory ahead of those in the system directory
|
||||
*/
|
||||
String tmpFontPath = jreFontDirName+File.pathSeparator+fontPath;
|
||||
StringTokenizer parser = new StringTokenizer(tmpFontPath,
|
||||
File.pathSeparator);
|
||||
|
||||
boolean found = false;
|
||||
try {
|
||||
while (!found && parser.hasMoreTokens()) {
|
||||
String newPath = parser.nextToken();
|
||||
File theFile = new File(newPath, fontFileName);
|
||||
if (theFile.canRead()) {
|
||||
found = true;
|
||||
String path = theFile.getAbsolutePath();
|
||||
if (defer) {
|
||||
registerDeferredFont(fontFileName, path,
|
||||
nativeNames,
|
||||
fontFormat, true,
|
||||
fontRank);
|
||||
} else {
|
||||
registerFontFile(path, nativeNames,
|
||||
fontFormat, true,
|
||||
fontRank);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
} catch (NoSuchElementException e) {
|
||||
System.err.println(e);
|
||||
}
|
||||
if (!found) {
|
||||
addToMissingFontFileList(fontFileName);
|
||||
}
|
||||
}
|
||||
|
||||
@Override
|
||||
protected FontConfiguration createFontConfiguration() {
|
||||
|
||||
FontConfiguration fc = new WFontConfiguration(this);
|
||||
fc.init();
|
||||
return fc;
|
||||
}
|
||||
|
||||
@Override
|
||||
public FontConfiguration createFontConfiguration(boolean preferLocaleFonts,
|
||||
boolean preferPropFonts) {
|
||||
|
||||
return new WFontConfiguration(this,
|
||||
preferLocaleFonts,preferPropFonts);
|
||||
}
|
||||
|
||||
protected void
|
||||
populateFontFileNameMap(HashMap<String,String> fontToFileMap,
|
||||
HashMap<String,String> fontToFamilyNameMap,
|
||||
HashMap<String,ArrayList<String>>
|
||||
familyToFontListMap,
|
||||
Locale locale) {
|
||||
|
||||
populateFontFileNameMap0(fontToFileMap, fontToFamilyNameMap,
|
||||
familyToFontListMap, locale);
|
||||
|
||||
}
|
||||
|
||||
private static native void
|
||||
populateFontFileNameMap0(HashMap<String,String> fontToFileMap,
|
||||
HashMap<String,String> fontToFamilyNameMap,
|
||||
HashMap<String,ArrayList<String>>
|
||||
familyToFontListMap,
|
||||
Locale locale);
|
||||
|
||||
public synchronized native String getFontPath(boolean noType1Fonts);
|
||||
|
||||
public String[] getDefaultPlatformFont() {
|
||||
|
||||
if (defaultPlatformFont != null) {
|
||||
return defaultPlatformFont;
|
||||
}
|
||||
|
||||
String[] info = new String[2];
|
||||
info[0] = "Arial";
|
||||
info[1] = "c:\\windows\\fonts";
|
||||
final String[] dirs = getPlatformFontDirs(true);
|
||||
if (dirs.length > 1) {
|
||||
String dir = (String)
|
||||
AccessController.doPrivileged(new PrivilegedAction() {
|
||||
public Object run() {
|
||||
for (int i=0; i<dirs.length; i++) {
|
||||
String path =
|
||||
dirs[i] + File.separator + "arial.ttf";
|
||||
File file = new File(path);
|
||||
if (file.exists()) {
|
||||
return dirs[i];
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
});
|
||||
if (dir != null) {
|
||||
info[1] = dir;
|
||||
}
|
||||
} else {
|
||||
info[1] = dirs[0];
|
||||
}
|
||||
info[1] = info[1] + File.separator + "arial.ttf";
|
||||
defaultPlatformFont = info;
|
||||
return defaultPlatformFont;
|
||||
}
|
||||
|
||||
/* register only TrueType/OpenType fonts
|
||||
* Because these need to be registed just for use when printing,
|
||||
* we defer the actual registration and the static initialiser
|
||||
* for the printing class makes the call to registerJREFontsForPrinting()
|
||||
*/
|
||||
static String fontsForPrinting = null;
|
||||
protected void registerJREFontsWithPlatform(String pathName) {
|
||||
fontsForPrinting = pathName;
|
||||
}
|
||||
|
||||
public static void registerJREFontsForPrinting() {
|
||||
final String pathName;
|
||||
synchronized (Win32GraphicsEnvironment.class) {
|
||||
GraphicsEnvironment.getLocalGraphicsEnvironment();
|
||||
if (fontsForPrinting == null) {
|
||||
return;
|
||||
}
|
||||
pathName = fontsForPrinting;
|
||||
fontsForPrinting = null;
|
||||
}
|
||||
java.security.AccessController.doPrivileged(
|
||||
new java.security.PrivilegedAction() {
|
||||
public Object run() {
|
||||
File f1 = new File(pathName);
|
||||
String[] ls = f1.list(SunFontManager.getInstance().
|
||||
getTrueTypeFilter());
|
||||
if (ls == null) {
|
||||
return null;
|
||||
}
|
||||
for (int i=0; i <ls.length; i++ ) {
|
||||
File fontFile = new File(f1, ls[i]);
|
||||
registerFontWithPlatform(fontFile.getAbsolutePath());
|
||||
}
|
||||
return null;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
protected static native void registerFontWithPlatform(String fontName);
|
||||
|
||||
protected static native void deRegisterFontWithPlatform(String fontName);
|
||||
|
||||
}
|
Loading…
Reference in New Issue
Block a user