/* * Copyright (c) 2011, 2013, 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.font; import com.sun.javafx.scene.text.GlyphList; import com.sun.javafx.geom.transform.Affine2D; import com.sun.javafx.geom.transform.BaseTransform; import com.sun.javafx.geom.Path2D; import com.sun.javafx.geom.Point2D; import com.sun.javafx.geom.Shape; public class CompositeStrike implements FontStrike { private CompositeFontResource fontResource; private float size; private int aaMode; BaseTransform transform; private FontStrike slot0Strike; private FontStrike[] strikeSlots; private FontStrikeDesc desc; DisposerRecord disposer; public void clearDesc() { fontResource.getStrikeMap().remove(desc); // For a composite strike, you also need to remove the strike // refs of the raw fonts. At the least this needs to remove // the slot 0 strike, but it may be that the fallback strikes // should be left alone as they could be shared. This needs // to be re-visited. if (slot0Strike != null) { slot0Strike.clearDesc(); } if (strikeSlots != null) { for (int i=1; i= strikeSlots.length) { FontStrike[] tmp = new FontStrike[fontResource.getNumSlots()]; System.arraycopy(strikeSlots, 0, tmp, 0, strikeSlots.length); strikeSlots = tmp; } if (strikeSlots[slot] == null) { FontResource slotResource = fontResource.getSlotResource(slot); strikeSlots[slot] = slotResource.getStrike(size, transform, getAAMode()); } return strikeSlots[slot]; } } public FontResource getFontResource() { return fontResource; } public int getStrikeSlotForGlyph(int glyphCode) { return (glyphCode >>> 24); } public float getSize() { return size; } public boolean drawAsShapes() { return getStrikeSlot(0).drawAsShapes(); } private PrismMetrics metrics; public Metrics getMetrics() { if (metrics == null) { PrismFontFile fr = (PrismFontFile)fontResource.getSlotResource(0); metrics = fr.getFontMetrics(size); } return metrics; } public Glyph getGlyph(char symbol) { int glyphCode = fontResource.getGlyphMapper().charToGlyph(symbol); return getGlyph(glyphCode); } public Glyph getGlyph(int glyphCode) { int slot = (glyphCode >>> 24); int slotglyphCode = glyphCode & CompositeGlyphMapper.GLYPHMASK; return getStrikeSlot(slot).getGlyph(slotglyphCode); } /** * Access to individual character advances are frequently needed for layout * understand that advance may vary for single glyph if ligatures or kerning * are enabled * @param ch char * @return advance of single char */ public float getCharAdvance(char ch) { int glyphCode = fontResource.getGlyphMapper().charToGlyph((int)ch); return fontResource.getAdvance(glyphCode, size); } @Override public int getQuantizedPosition(Point2D point) { return getStrikeSlot(0).getQuantizedPosition(point); } public Shape getOutline(GlyphList gl, BaseTransform transform) { Path2D result = new Path2D(); getOutline(gl, transform, result); return result; } void getOutline(GlyphList gl, BaseTransform transform, Path2D p) { p.reset(); if (gl == null) { return; } if (transform == null) { transform = BaseTransform.IDENTITY_TRANSFORM; } Affine2D t = new Affine2D(); for (int i = 0; i < gl.getGlyphCount(); i++) { int glyphCode = gl.getGlyphCode(i); if (glyphCode != CharToGlyphMapper.INVISIBLE_GLYPH_ID) { Glyph glyph = getGlyph(glyphCode); Shape gp = glyph.getShape(); if (gp != null) { t.setTransform(transform); t.translate(gl.getPosX(i), gl.getPosY(i)); p.append(gp.getPathIterator(t), false); } } } } }