/* * Copyright (c) 2011, 2016, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ package com.sun.javafx.webkit.prism; import com.sun.javafx.font.CharToGlyphMapper; import com.sun.javafx.font.FontFactory; import com.sun.javafx.font.FontResource; import com.sun.javafx.font.FontStrike; import com.sun.javafx.font.PGFont; import com.sun.javafx.geom.BaseBounds; import com.sun.javafx.geom.transform.BaseTransform; import com.sun.javafx.logging.PlatformLogger; import com.sun.javafx.logging.PlatformLogger.Level; import com.sun.javafx.scene.text.GlyphList; import com.sun.javafx.scene.text.TextLayout; import com.sun.javafx.text.TextRun; import static com.sun.javafx.webkit.prism.TextUtilities.getLayoutWidth; import static com.sun.javafx.webkit.prism.TextUtilities.getLayoutBounds; import com.sun.prism.GraphicsPipeline; import com.sun.webkit.graphics.WCFont; import com.sun.webkit.graphics.WCGlyphBuffer; import java.util.HashMap; final class WCFontImpl extends WCFont { private final static PlatformLogger log = PlatformLogger.getLogger(WCFontImpl.class.getName()); private static final HashMap FONT_MAP = new HashMap(); static WCFont getFont(String name, boolean bold, boolean italic, float size) { FontFactory factory = GraphicsPipeline.getPipeline().getFontFactory(); synchronized (FONT_MAP) { if (FONT_MAP.isEmpty()) { FONT_MAP.put("serif", "Serif"); FONT_MAP.put("dialog", "SansSerif"); FONT_MAP.put("helvetica", "SansSerif"); FONT_MAP.put("sansserif", "SansSerif"); FONT_MAP.put("sans-serif", "SansSerif"); FONT_MAP.put("monospace", "Monospaced"); FONT_MAP.put("monospaced", "Monospaced"); for (String family : factory.getFontFamilyNames()) { FONT_MAP.put(family.toLowerCase(), family); } } } String family = FONT_MAP.get(name.toLowerCase()); if (log.isLoggable(Level.FINE)) { StringBuilder sb = new StringBuilder("WCFontImpl.get("); sb.append(name).append(", ").append(size); if (bold) { sb.append(", bold"); } if (italic) { sb.append(", italic"); } log.fine(sb.append(") = ").append(family).toString()); } return (family != null) ? new WCFontImpl(factory.createFont(family, bold, italic, size)) : null; } private final PGFont font; WCFontImpl(PGFont font) { this.font = font; } @Override public WCFont deriveFont(float size) { FontFactory factory = GraphicsPipeline.getPipeline().getFontFactory(); return new WCFontImpl( factory.deriveFont(font, font.getFontResource().isBold(), font.getFontResource().isItalic(), size)); } @Override public int getOffsetForPosition(String str, float x) { TextLayout layout = TextUtilities.createLayout(str, font); GlyphList[] runs = layout.getRuns(); TextRun run = (TextRun) runs[0]; int offset = run.getOffsetAtX(x, null); if (log.isLoggable(Level.FINE)) { log.fine(String.format("str='%s' (length=%d), x=%.2f => %d", str, str.length(), x, offset)); } return offset; } @Override public WCGlyphBuffer getGlyphsAndAdvances(String str, int from, int to, boolean rtl) { if (log.isLoggable(Level.FINE)) { log.fine(String.format( "str='%s' (length=%d), from=%d, to=%d, rtl=%b", str, str.length(), from, to, rtl)); } TextLayout layout = TextUtilities.createLayout( str.substring(from, to), getPlatformFont()); int count = 0; GlyphList[] runs = layout.getRuns(); for (GlyphList run: runs) { count += run.getGlyphCount(); } int[] glyphs = new int[count]; float[] adv = new float[count]; count = 0; for (GlyphList run: layout.getRuns()) { int gc = run.getGlyphCount(); for (int i = 0; i < gc; i++) { glyphs[count] = run.getGlyphCode(i); adv[count] = run.getPosX(i + 1) - run.getPosX(i); count++; } } // inital advance (see RT-29908) float x = 0; if (rtl) { x += (TextUtilities.getLayoutWidth(str.substring(from), getPlatformFont()) - layout.getBounds().getWidth()); } else { x += TextUtilities.getLayoutWidth(str.substring(0, from), getPlatformFont()); } return new WCGlyphBuffer(glyphs, adv, x); } private FontStrike strike; private FontStrike getFontStrike() { if (strike == null) { strike = font.getStrike(BaseTransform.IDENTITY_TRANSFORM, FontResource.AA_LCD); } return strike; } @Override public double getGlyphWidth(int glyph) { return getFontStrike().getFontResource().getAdvance(glyph, font.getSize()); } @Override public float getXHeight() { return getFontStrike().getMetrics().getXHeight(); } @Override public int[] getGlyphCodes(char[] chars) { int[] glyphs = new int[chars.length]; CharToGlyphMapper mapper = getFontStrike().getFontResource().getGlyphMapper(); mapper.charsToGlyphs(chars.length, chars, glyphs); return glyphs; } @Override public double getStringWidth(String str) { double result = getLayoutWidth(str, font); if (log.isLoggable(Level.FINE)) { log.fine(String.format("str='%s' (length=%d) => %.1f", str, str.length(), result)); } return result; } @Override public double[] getStringBounds(String str, int from, int to, boolean rtl) { float beforeWidth = getLayoutWidth(str.substring(0, from), font); BaseBounds bounds = getLayoutBounds(str.substring(0, to), font); double[] result = new double[] { beforeWidth, // see RTL case below 0, // not really used bounds.getWidth() - beforeWidth, bounds.getHeight(), // not really used }; if (rtl) { float totalWidth = getLayoutWidth(str, font); result[0] = totalWidth - bounds.getWidth(); } if (log.isLoggable(Level.FINE)) { log.fine(String.format( "str='%s' (length=%d) [%d, %d], rtl=%b => [%.1f, %.1f + %.1f x %.1f]", str, str.length(), from, to, rtl, result[0], result[1], result[2], result[3])); } return result; } public float getAscent() { // REMIND: This method needs to require a render context. float res = - getFontStrike().getMetrics().getAscent(); if (log.isLoggable(Level.FINER)) { log.finer("getAscent({0}, {1}) = {2}", new Object[] {font.getName(), font.getSize(), res}); } return res; } public float getDescent() { // REMIND: This method needs to require a render context. float res = getFontStrike().getMetrics().getDescent(); if (log.isLoggable(Level.FINER)) { log.finer("getDescent({0}, {1}) = {2}", new Object[] {font.getName(), font.getSize(), res}); } return res; } public float getLineSpacing() { // REMIND: This method needs to require a render context. float res = getFontStrike().getMetrics().getLineHeight(); if (log.isLoggable(Level.FINER)) { log.finer("getLineSpacing({0}, {1}) = {2}", new Object[] {font.getName(), font.getSize(), res}); } return res; } public float getLineGap() { // REMIND: This method needs to require a render context. float res = getFontStrike().getMetrics().getLineGap(); if (log.isLoggable(Level.FINER)) { log.finer("getLineGap({0}, {1}) = {2}", new Object[] {font.getName(), font.getSize(), res }); } return res; } public boolean hasUniformLineMetrics() { return false; } public Object getPlatformFont() { return font; } @Override public float getCapHeight() { return getFontStrike().getMetrics().getCapHeight(); } }