| /* |
| * Copyright (c) 2011, 2014, 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. |
| */ |
| |
| #import <JavaNativeFoundation/JavaNativeFoundation.h> |
| #import "java_awt_geom_PathIterator.h" |
| #import "sun_font_CStrike.h" |
| #import "sun_font_CStrikeDisposer.h" |
| #import "CGGlyphImages.h" |
| #import "CGGlyphOutlines.h" |
| #import "CoreTextSupport.h" |
| #include "fontscalerdefs.h" |
| |
| /* Use THIS_FILE when it is available. */ |
| #ifndef THIS_FILE |
| #define THIS_FILE __FILE__ |
| #endif |
| |
| @implementation AWTStrike |
| |
| static CGAffineTransform sInverseTX = { 1, 0, 0, -1, 0, 0 }; |
| |
| - (id) initWithFont:(AWTFont *)awtFont |
| tx:(CGAffineTransform)tx |
| invDevTx:(CGAffineTransform)invDevTx |
| style:(JRSFontRenderingStyle)style |
| aaStyle:(jint)aaStyle { |
| |
| self = [super init]; |
| if (self) { |
| fAWTFont = [awtFont retain]; |
| fStyle = style; |
| fAAStyle = aaStyle; |
| |
| fTx = tx; // composited glyph and device transform |
| |
| fAltTx = tx; |
| fAltTx.b *= -1; |
| fAltTx.d *= -1; |
| |
| invDevTx.b *= -1; |
| invDevTx.c *= -1; |
| fFontTx = CGAffineTransformConcat(CGAffineTransformConcat(tx, invDevTx), sInverseTX); |
| fDevTx = CGAffineTransformInvert(CGAffineTransformConcat(invDevTx, sInverseTX)); |
| |
| // the "font size" is the square root of the determinant of the matrix |
| fSize = sqrt(fabs(fFontTx.a * fFontTx.d - fFontTx.b * fFontTx.c)); |
| } |
| return self; |
| } |
| |
| - (void) dealloc { |
| [fAWTFont release]; |
| fAWTFont = nil; |
| |
| [super dealloc]; |
| } |
| |
| + (AWTStrike *) awtStrikeForFont:(AWTFont *)awtFont |
| tx:(CGAffineTransform)tx |
| invDevTx:(CGAffineTransform)invDevTx |
| style:(JRSFontRenderingStyle)style |
| aaStyle:(jint)aaStyle { |
| |
| return [[[AWTStrike alloc] initWithFont:awtFont |
| tx:tx invDevTx:invDevTx |
| style:style |
| aaStyle:aaStyle] autorelease]; |
| } |
| |
| @end |
| |
| |
| #define AWT_FONT_CLEANUP_SETUP \ |
| BOOL _fontThrowJavaException = NO; |
| |
| #define AWT_FONT_CLEANUP_CHECK(a) \ |
| if ((a) == NULL) { \ |
| _fontThrowJavaException = YES; \ |
| goto cleanup; \ |
| } \ |
| if ((*env)->ExceptionCheck(env) == JNI_TRUE) { \ |
| goto cleanup; \ |
| } |
| |
| #define AWT_FONT_CLEANUP_FINISH \ |
| if (_fontThrowJavaException == YES) { \ |
| char s[512]; \ |
| sprintf(s, "%s-%s:%d", THIS_FILE, __FUNCTION__, __LINE__); \ |
| [JNFException raise:env as:kRuntimeException reason:s]; \ |
| } |
| |
| |
| /* |
| * Creates an affine transform from the corresponding doubles sent |
| * from CStrike.getGlyphTx(). |
| */ |
| static inline CGAffineTransform |
| GetTxFromDoubles(JNIEnv *env, jdoubleArray txArray) |
| { |
| if (txArray == NULL) { |
| return CGAffineTransformIdentity; |
| } |
| |
| jdouble *txPtr = (*env)->GetPrimitiveArrayCritical(env, txArray, NULL); |
| if (txPtr == NULL) { |
| return CGAffineTransformIdentity; |
| } |
| |
| CGAffineTransform tx = |
| CGAffineTransformMake(txPtr[0], txPtr[1], txPtr[2], |
| txPtr[3], txPtr[4], txPtr[5]); |
| tx = CGAffineTransformConcat(sInverseTX, tx); |
| |
| (*env)->ReleasePrimitiveArrayCritical(env, txArray, txPtr, JNI_ABORT); |
| |
| return tx; |
| } |
| |
| /* |
| * Class: sun_font_CStrike |
| * Method: getNativeGlyphAdvance |
| * Signature: (JI)F |
| */ |
| JNIEXPORT jfloat JNICALL |
| Java_sun_font_CStrike_getNativeGlyphAdvance |
| (JNIEnv *env, jclass clazz, jlong awtStrikePtr, jint glyphCode) |
| { |
| CGSize advance; |
| JNF_COCOA_ENTER(env); |
| AWTStrike *awtStrike = (AWTStrike *)jlong_to_ptr(awtStrikePtr); |
| AWTFont *awtFont = awtStrike->fAWTFont; |
| |
| // negative glyph codes are really unicodes, which were placed there by the mapper |
| // to indicate we should use CoreText to substitute the character |
| CGGlyph glyph; |
| const CTFontRef fallback = CTS_CopyCTFallbackFontAndGlyphForJavaGlyphCode(awtFont, glyphCode, &glyph); |
| CTFontGetAdvancesForGlyphs(fallback, kCTFontDefaultOrientation, &glyph, &advance, 1); |
| CFRelease(fallback); |
| advance = CGSizeApplyAffineTransform(advance, awtStrike->fFontTx); |
| if (!JRSFontStyleUsesFractionalMetrics(awtStrike->fStyle)) { |
| advance.width = round(advance.width); |
| } |
| |
| JNF_COCOA_EXIT(env); |
| return advance.width; |
| } |
| |
| /* |
| * Class: sun_font_CStrike |
| * Method: getNativeGlyphImageBounds |
| * Signature: (JJILjava/awt/geom/Rectangle2D/Float;DD)V |
| */ |
| JNIEXPORT void JNICALL |
| Java_sun_font_CStrike_getNativeGlyphImageBounds |
| (JNIEnv *env, jclass clazz, |
| jlong awtStrikePtr, jint glyphCode, |
| jobject result /*Rectangle*/, jdouble x, jdouble y) |
| { |
| JNF_COCOA_ENTER(env); |
| |
| AWTStrike *awtStrike = (AWTStrike *)jlong_to_ptr(awtStrikePtr); |
| AWTFont *awtFont = awtStrike->fAWTFont; |
| |
| CGAffineTransform tx = awtStrike->fAltTx; |
| tx.tx += x; |
| tx.ty += y; |
| |
| // negative glyph codes are really unicodes, which were placed there by the mapper |
| // to indicate we should use CoreText to substitute the character |
| CGGlyph glyph; |
| const CTFontRef fallback = CTS_CopyCTFallbackFontAndGlyphForJavaGlyphCode(awtFont, glyphCode, &glyph); |
| |
| CGRect bbox; |
| JRSFontGetBoundingBoxesForGlyphsAndStyle(fallback, &tx, awtStrike->fStyle, &glyph, 1, &bbox); |
| CFRelease(fallback); |
| |
| // the origin of this bounding box is relative to the bottom-left corner baseline |
| CGFloat decender = -bbox.origin.y; |
| bbox.origin.y = -bbox.size.height + decender; |
| |
| // Rectangle2D.Float.setRect(float x, float y, float width, float height); |
| static JNF_CLASS_CACHE(sjc_Rectangle2D_Float, "java/awt/geom/Rectangle2D$Float"); // cache class id for Rectangle |
| static JNF_MEMBER_CACHE(sjr_Rectangle2DFloat_setRect, sjc_Rectangle2D_Float, "setRect", "(FFFF)V"); |
| JNFCallVoidMethod(env, result, sjr_Rectangle2DFloat_setRect, (jfloat)bbox.origin.x, (jfloat)bbox.origin.y, (jfloat)bbox.size.width, (jfloat)bbox.size.height); |
| |
| JNF_COCOA_EXIT(env); |
| } |
| |
| /* |
| * Class: sun_font_CStrike |
| * Method: getNativeGlyphOutline |
| * Signature: (JJIDD)Ljava/awt/geom/GeneralPath; |
| */ |
| JNIEXPORT jobject JNICALL |
| Java_sun_font_CStrike_getNativeGlyphOutline |
| (JNIEnv *env, jclass clazz, |
| jlong awtStrikePtr, jint glyphCode, jdouble xPos, jdouble yPos) |
| { |
| jobject generalPath = NULL; |
| |
| JNF_COCOA_ENTER(env); |
| |
| AWTPathRef path = NULL; |
| jfloatArray pointCoords = NULL; |
| jbyteArray pointTypes = NULL; |
| |
| AWT_FONT_CLEANUP_SETUP; |
| |
| AWTStrike *awtStrike = (AWTStrike *)jlong_to_ptr(awtStrikePtr); |
| AWTFont *awtfont = awtStrike->fAWTFont; |
| |
| AWT_FONT_CLEANUP_CHECK(awtfont); |
| |
| // inverting the shear order and sign to compensate for the flipped coordinate system |
| CGAffineTransform tx = awtStrike->fTx; |
| tx.tx += xPos; |
| tx.ty += yPos; |
| |
| // get the right font and glyph for this "Java GlyphCode" |
| |
| CGGlyph glyph; |
| const CTFontRef font = CTS_CopyCTFallbackFontAndGlyphForJavaGlyphCode(awtfont, glyphCode, &glyph); |
| |
| // get the advance of this glyph |
| CGSize advance; |
| CTFontGetAdvancesForGlyphs(font, kCTFontDefaultOrientation, &glyph, &advance, 1); |
| |
| // Create AWTPath |
| path = AWTPathCreate(CGSizeMake(xPos, yPos)); |
| AWT_FONT_CLEANUP_CHECK(path); |
| |
| // Get the paths |
| tx = awtStrike->fTx; |
| tx = CGAffineTransformConcat(tx, sInverseTX); |
| AWTGetGlyphOutline(&glyph, (NSFont *)font, &advance, &tx, 0, 1, &path); |
| CFRelease(font); |
| |
| pointCoords = (*env)->NewFloatArray(env, path->fNumberOfDataElements); |
| AWT_FONT_CLEANUP_CHECK(pointCoords); |
| |
| (*env)->SetFloatArrayRegion(env, pointCoords, 0, path->fNumberOfDataElements, (jfloat*)path->fSegmentData); |
| |
| // Copy the pointTypes to the general path |
| pointTypes = (*env)->NewByteArray(env, path->fNumberOfSegments); |
| AWT_FONT_CLEANUP_CHECK(pointTypes); |
| |
| (*env)->SetByteArrayRegion(env, pointTypes, 0, path->fNumberOfSegments, (jbyte*)path->fSegmentType); |
| |
| static JNF_CLASS_CACHE(jc_GeneralPath, "java/awt/geom/GeneralPath"); |
| static JNF_CTOR_CACHE(jc_GeneralPath_ctor, jc_GeneralPath, "(I[BI[FI)V"); |
| generalPath = JNFNewObject(env, jc_GeneralPath_ctor, java_awt_geom_PathIterator_WIND_NON_ZERO, pointTypes, path->fNumberOfSegments, pointCoords, path->fNumberOfDataElements); // AWT_THREADING Safe (known object) |
| |
| // Cleanup |
| cleanup: |
| if (path != NULL) { |
| AWTPathFree(path); |
| path = NULL; |
| } |
| |
| if (pointCoords != NULL) { |
| (*env)->DeleteLocalRef(env, pointCoords); |
| pointCoords = NULL; |
| } |
| |
| if (pointTypes != NULL) { |
| (*env)->DeleteLocalRef(env, pointTypes); |
| pointTypes = NULL; |
| } |
| |
| AWT_FONT_CLEANUP_FINISH; |
| JNF_COCOA_EXIT(env); |
| return generalPath; |
| } |
| |
| /* |
| * Class: sun_font_CStrike |
| * Method: getGlyphImagePtrsNative |
| * Signature: (JJ[J[II)V |
| */ |
| JNIEXPORT void JNICALL |
| Java_sun_font_CStrike_getGlyphImagePtrsNative |
| (JNIEnv *env, jclass clazz, |
| jlong awtStrikePtr, jlongArray glyphInfoLongArray, |
| jintArray glyphCodes, jint len) |
| { |
| JNF_COCOA_ENTER(env); |
| |
| AWTStrike *awtStrike = (AWTStrike *)jlong_to_ptr(awtStrikePtr); |
| |
| jlong *glyphInfos = |
| (*env)->GetPrimitiveArrayCritical(env, glyphInfoLongArray, NULL); |
| |
| jint *rawGlyphCodes = |
| (*env)->GetPrimitiveArrayCritical(env, glyphCodes, NULL); |
| @try { |
| if (rawGlyphCodes != NULL && glyphInfos != NULL) { |
| CGGlyphImages_GetGlyphImagePtrs(glyphInfos, awtStrike, |
| rawGlyphCodes, len); |
| } |
| } |
| @finally { |
| if (rawGlyphCodes != NULL) { |
| (*env)->ReleasePrimitiveArrayCritical(env, glyphCodes, |
| rawGlyphCodes, JNI_ABORT); |
| } |
| if (glyphInfos != NULL) { |
| // Do not use JNI_COMMIT, as that will not free the buffer copy |
| // when +ProtectJavaHeap is on. |
| (*env)->ReleasePrimitiveArrayCritical(env, glyphInfoLongArray, |
| glyphInfos, 0); |
| } |
| } |
| |
| JNF_COCOA_EXIT(env); |
| } |
| |
| /* |
| * Class: sun_font_CStrike |
| * Method: createNativeStrikePtr |
| * Signature: (J[D[DII)J |
| */ |
| JNIEXPORT jlong JNICALL Java_sun_font_CStrike_createNativeStrikePtr |
| (JNIEnv *env, jclass clazz, jlong nativeFontPtr, jdoubleArray glyphTxArray, jdoubleArray invDevTxArray, jint aaStyle, jint fmHint) |
| { |
| AWTStrike *awtStrike = nil; |
| JNF_COCOA_ENTER(env); |
| |
| AWTFont *awtFont = (AWTFont *)jlong_to_ptr(nativeFontPtr); |
| JRSFontRenderingStyle style = JRSFontGetRenderingStyleForHints(fmHint, aaStyle); |
| |
| CGAffineTransform glyphTx = GetTxFromDoubles(env, glyphTxArray); |
| CGAffineTransform invDevTx = GetTxFromDoubles(env, invDevTxArray); |
| |
| awtStrike = [AWTStrike awtStrikeForFont:awtFont tx:glyphTx invDevTx:invDevTx style:style aaStyle:aaStyle]; // autoreleased |
| |
| if (awtStrike) |
| { |
| CFRetain(awtStrike); // GC |
| } |
| |
| JNF_COCOA_EXIT(env); |
| return ptr_to_jlong(awtStrike); |
| } |
| |
| /* |
| * Class: sun_font_CStrike |
| * Method: disposeNativeStrikePtr |
| * Signature: (J)V |
| */ |
| JNIEXPORT void JNICALL |
| Java_sun_font_CStrike_disposeNativeStrikePtr |
| (JNIEnv *env, jclass clazz, jlong awtStrike) |
| { |
| JNF_COCOA_ENTER(env); |
| |
| if (awtStrike) { |
| CFRelease((AWTStrike *)jlong_to_ptr(awtStrike)); // GC |
| } |
| |
| JNF_COCOA_EXIT(env); |
| } |
| |
| /* |
| * Class: sun_font_CStrike |
| * Method: getFontMetrics |
| * Signature: (J)Lsun/font/StrikeMetrics; |
| */ |
| JNIEXPORT jobject JNICALL |
| Java_sun_font_CStrike_getFontMetrics |
| (JNIEnv *env, jclass clazz, jlong awtStrikePtr) |
| { |
| jobject metrics = NULL; |
| |
| JNF_COCOA_ENTER(env); |
| AWT_FONT_CLEANUP_SETUP; |
| |
| AWTFont *awtfont = ((AWTStrike *)jlong_to_ptr(awtStrikePtr))->fAWTFont; |
| AWT_FONT_CLEANUP_CHECK(awtfont); |
| |
| CGFontRef cgFont = awtfont->fNativeCGFont; |
| |
| jfloat ay=0.0, dy=0.0, mx=0.0, ly=0.0; |
| int unitsPerEm = CGFontGetUnitsPerEm(cgFont); |
| CGFloat scaleX = (1.0 / unitsPerEm); |
| CGFloat scaleY = (1.0 / unitsPerEm); |
| |
| // Ascent |
| ay = -(CGFloat)CGFontGetAscent(cgFont) * scaleY; |
| |
| // Descent |
| dy = -(CGFloat)CGFontGetDescent(cgFont) * scaleY; |
| |
| // Leading |
| ly = (CGFloat)CGFontGetLeading(cgFont) * scaleY; |
| |
| // Max Advance for Font Direction (Strictly horizontal) |
| mx = [awtfont->fFont maximumAdvancement].width; |
| |
| /* |
| * ascent: no need to set ascentX - it will be zero. |
| * descent: no need to set descentX - it will be zero. |
| * baseline: old releases "made up" a number and also seemed to |
| * make it up for "X" and set "Y" to 0. |
| * leadingX: no need to set leadingX - it will be zero. |
| * leadingY: made-up number, but being compatible with what 1.4.x did. |
| * advance: no need to set yMaxLinearAdvanceWidth - it will be zero. |
| */ |
| |
| JNF_CLASS_CACHE(sjc_StrikeMetrics, "sun/font/StrikeMetrics"); |
| JNF_CTOR_CACHE(strikeMetricsCtr, sjc_StrikeMetrics, "(FFFFFFFFFF)V"); |
| metrics = JNFNewObject(env, strikeMetricsCtr, |
| 0.0, ay, 0.0, dy, 1.0, |
| 0.0, 0.0, ly, mx, 0.0); |
| |
| cleanup: |
| AWT_FONT_CLEANUP_FINISH; |
| JNF_COCOA_EXIT(env); |
| |
| return metrics; |
| } |
| |
| extern void AccelGlyphCache_RemoveAllInfos(GlyphInfo* glyph); |
| /* |
| * Class: sun_font_CStrikeDisposer |
| * Method: removeGlyphInfoFromCache |
| * Signature: (J)V |
| */ |
| JNIEXPORT void JNICALL Java_sun_font_CStrikeDisposer_removeGlyphInfoFromCache |
| (JNIEnv *env, jclass cls, jlong glyphInfo) |
| { |
| JNF_COCOA_ENTER(env); |
| |
| AccelGlyphCache_RemoveAllCellInfos((GlyphInfo*)jlong_to_ptr(glyphInfo)); |
| |
| JNF_COCOA_EXIT(env); |
| } |