com.sun.javafx.font.CompositeStrike Maven / Gradle / Ivy
/*
* Copyright (c) 2011, 2023, 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;
@Override
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[slot+1];
System.arraycopy(strikeSlots, 0, tmp, 0, strikeSlots.length);
strikeSlots = tmp;
}
if (strikeSlots[slot] == null) {
FontResource slotResource = fontResource.getSlotResource(slot);
if (slotResource != null) {
strikeSlots[slot] = slotResource.getStrike(size, transform,
getAAMode());
}
}
return strikeSlots[slot];
}
}
@Override
public FontResource getFontResource() {
return fontResource;
}
public int getStrikeSlotForGlyph(int glyphCode) {
return (glyphCode >>> 24);
}
@Override
public float getSize() {
return size;
}
@Override
public boolean drawAsShapes() {
return getStrikeSlot(0).drawAsShapes();
}
private PrismMetrics metrics;
@Override
public Metrics getMetrics() {
if (metrics == null) {
PrismFontFile fr = (PrismFontFile)fontResource.getSlotResource(0);
metrics = fr.getFontMetrics(size);
}
return metrics;
}
@Override
public Glyph getGlyph(char symbol) {
int glyphCode = fontResource.getGlyphMapper().charToGlyph(symbol);
return getGlyph(glyphCode);
}
@Override
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
*/
@Override
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);
}
@Override
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);
}
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy