Paint.cpp revision 57c4fed236cbec0944824b534fb9c828ac0d1f81
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* libs/android_runtime/android/graphics/Paint.cpp
29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**
39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** Copyright 2006, The Android Open Source Project
49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**
58451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes** Licensed under the Apache License, Version 2.0 (the "License");
68451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes** you may not use this file except in compliance with the License.
78451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes** You may obtain a copy of the License at
89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**
98451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes**     http://www.apache.org/licenses/LICENSE-2.0
109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project**
118451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes** Unless required by applicable law or agreed to in writing, software
128451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes** distributed under the License is distributed on an "AS IS" BASIS,
138451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
148451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes** See the License for the specific language governing permissions and
159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project** limitations under the License.
169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project*/
179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
18f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn#define LOG_TAG "Paint"
19f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn
20f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn#include <utils/Log.h>
21f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn
229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "jni.h"
239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "GraphicsJNI.h"
249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include <android_runtime/AndroidRuntime.h>
25ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett#include <ScopedUtfChars.h>
269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SkBlurDrawLooper.h"
289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SkColorFilter.h"
299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SkMaskFilter.h"
309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SkRasterizer.h"
319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SkShader.h"
329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SkTypeface.h"
339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project#include "SkXfermode.h"
34ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett#include "unicode/uloc.h"
350c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt#include "unicode/ushape.h"
36c29a0a4664a4b9871fadd668b632469a0db240b9Derek Sollenberger#include "utils/Blur.h"
370c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
38e368b6bcfd499543a876b4cae02f1ca563ff1cb6Raph Levien#include <minikin/GraphemeBreak.h>
391a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien#include "MinikinSkia.h"
401a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien#include "MinikinUtils.h"
416ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod#include "Paint.h"
423d528c403c8945cc326bb656ff0c2e11373401daRaph Levien#include "TypefaceImpl.h"
431a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien
440c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt// temporary for debugging
455c13d89c1332fcc499379b9064b891187b75ca32Chet Haase#include <Caches.h>
460c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt#include <utils/Log.h>
479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectnamespace android {
499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstruct JMetricsID {
519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    jfieldID    top;
529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    jfieldID    ascent;
539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    jfieldID    descent;
549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    jfieldID    bottom;
559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    jfieldID    leading;
569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project};
579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jclass   gFontMetrics_class;
599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic JMetricsID gFontMetrics_fieldID;
609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jclass   gFontMetricsInt_class;
629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic JMetricsID gFontMetricsInt_fieldID;
639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
64ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenbergerstatic jclass   gPaint_class;
65ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenbergerstatic jfieldID gPaint_nativeInstanceID;
66ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenbergerstatic jfieldID gPaint_nativeTypefaceID;
67ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger
686ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbodstatic void defaultSettingsForAndroid(Paint* paint) {
69c511bee87cda99a252d1a62487f47c8f05aee78cFabrice Di Meglio    // GlyphID encoding is required because we are using Harfbuzz shaping
706ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    paint->setTextEncoding(Paint::kGlyphID_TextEncoding);
713d63e0119dc763ed0a06fd7498375746fd391d80Mike Reed}
723d63e0119dc763ed0a06fd7498375746fd391d80Mike Reed
736ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbodclass PaintGlue {
749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic:
750c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt    enum MoveOpt {
760c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        AFTER, AT_OR_AFTER, BEFORE, AT_OR_BEFORE, AT
770c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt    };
789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
79ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    static Paint* getNativePaint(JNIEnv* env, jobject paint) {
80ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        SkASSERT(env);
81ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        SkASSERT(paint);
82ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        SkASSERT(env->IsInstanceOf(paint, gPaint_class));
83ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        jlong paintHandle = env->GetLongField(paint, gPaint_nativeInstanceID);
84ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        android::Paint* p = reinterpret_cast<android::Paint*>(paintHandle);
85ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        SkASSERT(p);
86ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        return p;
87ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    }
88ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger
89ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    static TypefaceImpl* getNativeTypeface(JNIEnv* env, jobject paint) {
90ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        SkASSERT(env);
91ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        SkASSERT(paint);
92ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        SkASSERT(env->IsInstanceOf(paint, gPaint_class));
93ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        jlong typefaceHandle = env->GetLongField(paint, gPaint_nativeTypefaceID);
94ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        android::TypefaceImpl* p = reinterpret_cast<android::TypefaceImpl*>(typefaceHandle);
95ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        return p;
96ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    }
97ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger
9836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static void finalizer(JNIEnv* env, jobject clazz, jlong objHandle) {
996ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
1009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        delete obj;
1019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong init(JNIEnv* env, jobject clazz) {
1046ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = new Paint();
1053d63e0119dc763ed0a06fd7498375746fd391d80Mike Reed        defaultSettingsForAndroid(obj);
10636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return reinterpret_cast<jlong>(obj);
1079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong initWithPaint(JNIEnv* env, jobject clazz, jlong paintHandle) {
1106ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
1116ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = new Paint(*paint);
11236bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return reinterpret_cast<jlong>(obj);
1139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1148451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
11536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static void reset(JNIEnv* env, jobject clazz, jlong objHandle) {
1166ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
1179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        obj->reset();
1183d63e0119dc763ed0a06fd7498375746fd391d80Mike Reed        defaultSettingsForAndroid(obj);
1199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1208451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
12136bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static void assign(JNIEnv* env, jobject clazz, jlong dstPaintHandle, jlong srcPaintHandle) {
1226ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* dst = reinterpret_cast<Paint*>(dstPaintHandle);
1236ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        const Paint* src = reinterpret_cast<Paint*>(srcPaintHandle);
1249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        *dst = *src;
1259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1268451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
1274c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed    // Equivalent to the Java Paint's FILTER_BITMAP_FLAG.
1284c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed    static const uint32_t sFilterBitmapFlag = 0x02;
1294c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed
1309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jint getFlags(JNIEnv* env, jobject paint) {
1319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
132ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        Paint* nativePaint = getNativePaint(env, paint);
1334c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        uint32_t result = nativePaint->getFlags();
1344c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        result &= ~sFilterBitmapFlag; // Filtering no longer stored in this bit. Mask away.
1356ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        if (nativePaint->getFilterLevel() != Paint::kNone_FilterLevel) {
1364c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed            result |= sFilterBitmapFlag;
1374c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        }
13836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return static_cast<jint>(result);
1399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1408451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
1419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setFlags(JNIEnv* env, jobject paint, jint flags) {
1429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
143ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        Paint* nativePaint = getNativePaint(env, paint);
1444c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        // Instead of modifying 0x02, change the filter level.
1454c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        nativePaint->setFilterLevel(flags & sFilterBitmapFlag
1466ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod                ? Paint::kLow_FilterLevel
1476ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod                : Paint::kNone_FilterLevel);
1484c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        // Don't pass through filter flag, which is no longer stored in paint's flags.
1494c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        flags &= ~sFilterBitmapFlag;
1504c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        // Use the existing value for 0x02.
1514c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        const uint32_t existing0x02Flag = nativePaint->getFlags() & sFilterBitmapFlag;
1524c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        flags |= existing0x02Flag;
1534c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        nativePaint->setFlags(flags);
1549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1558451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
156f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn    static jint getHinting(JNIEnv* env, jobject paint) {
157f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn        NPE_CHECK_RETURN_ZERO(env, paint);
158ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        return getNativePaint(env, paint)->getHinting()
1596ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod                == Paint::kNo_Hinting ? 0 : 1;
160f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn    }
161f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn
162f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn    static void setHinting(JNIEnv* env, jobject paint, jint mode) {
163f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn        NPE_CHECK_RETURN_VOID(env, paint);
164ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setHinting(
1656ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod                mode == 0 ? Paint::kNo_Hinting : Paint::kNormal_Hinting);
166f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn    }
167f43fa5746ee5b81a6e386d36594094d079ac8160Dianne Hackborn
1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setAntiAlias(JNIEnv* env, jobject paint, jboolean aa) {
1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
170ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setAntiAlias(aa);
1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1728451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setLinearText(JNIEnv* env, jobject paint, jboolean linearText) {
1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
175ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setLinearText(linearText);
1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1778451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setSubpixelText(JNIEnv* env, jobject paint, jboolean subpixelText) {
1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
180ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setSubpixelText(subpixelText);
1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1828451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setUnderlineText(JNIEnv* env, jobject paint, jboolean underlineText) {
1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
185ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setUnderlineText(underlineText);
1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1878451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setStrikeThruText(JNIEnv* env, jobject paint, jboolean strikeThruText) {
1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
190ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setStrikeThruText(strikeThruText);
1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1928451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setFakeBoldText(JNIEnv* env, jobject paint, jboolean fakeBoldText) {
1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
195ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setFakeBoldText(fakeBoldText);
1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
1978451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setFilterBitmap(JNIEnv* env, jobject paint, jboolean filterBitmap) {
1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
200ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setFilterLevel(
2016ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod                filterBitmap ? Paint::kLow_FilterLevel : Paint::kNone_FilterLevel);
2029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2038451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setDither(JNIEnv* env, jobject paint, jboolean dither) {
2059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
206ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setDither(dither);
2079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2088451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
20936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jint getStyle(JNIEnv* env, jobject clazz,jlong objHandle) {
2106ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
21136bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return static_cast<jint>(obj->getStyle());
2129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2138451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
21436bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static void setStyle(JNIEnv* env, jobject clazz, jlong objHandle, jint styleHandle) {
2156ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
2166ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::Style style = static_cast<Paint::Style>(styleHandle);
2179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        obj->setStyle(style);
2189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2198451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jint getColor(JNIEnv* env, jobject paint) {
2219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
22236bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        int color;
223ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        color = getNativePaint(env, paint)->getColor();
22436bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return static_cast<jint>(color);
2259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2268451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jint getAlpha(JNIEnv* env, jobject paint) {
2289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
22936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        int alpha;
230ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        alpha = getNativePaint(env, paint)->getAlpha();
23136bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return static_cast<jint>(alpha);
2329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2338451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setColor(JNIEnv* env, jobject paint, jint color) {
2359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
236ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setColor(color);
2379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2388451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setAlpha(JNIEnv* env, jobject paint, jint a) {
2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
241ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setAlpha(a);
2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2438451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jfloat getStrokeWidth(JNIEnv* env, jobject paint) {
2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
246ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        return SkScalarToFloat(getNativePaint(env, paint)->getStrokeWidth());
2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2488451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setStrokeWidth(JNIEnv* env, jobject paint, jfloat width) {
2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
251ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setStrokeWidth(width);
2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2538451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jfloat getStrokeMiter(JNIEnv* env, jobject paint) {
2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
256ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        return SkScalarToFloat(getNativePaint(env, paint)->getStrokeMiter());
2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2588451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setStrokeMiter(JNIEnv* env, jobject paint, jfloat miter) {
2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
261ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setStrokeMiter(miter);
2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2638451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
26436bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jint getStrokeCap(JNIEnv* env, jobject clazz, jlong objHandle) {
2656ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
26636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return static_cast<jint>(obj->getStrokeCap());
2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2688451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
26936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static void setStrokeCap(JNIEnv* env, jobject clazz, jlong objHandle, jint capHandle) {
2706ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
2716ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::Cap cap = static_cast<Paint::Cap>(capHandle);
2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        obj->setStrokeCap(cap);
2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2748451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
27536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jint getStrokeJoin(JNIEnv* env, jobject clazz, jlong objHandle) {
2766ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
27736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return static_cast<jint>(obj->getStrokeJoin());
2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2798451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
28036bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static void setStrokeJoin(JNIEnv* env, jobject clazz, jlong objHandle, jint joinHandle) {
2816ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
2826ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::Join join = (Paint::Join) joinHandle;
2839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        obj->setStrokeJoin(join);
2849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2858451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
28636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jboolean getFillPath(JNIEnv* env, jobject clazz, jlong objHandle, jlong srcHandle, jlong dstHandle) {
2876ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
28836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkPath* src = reinterpret_cast<SkPath*>(srcHandle);
28936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkPath* dst = reinterpret_cast<SkPath*>(dstHandle);
29036bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return obj->getFillPath(*src, dst) ? JNI_TRUE : JNI_FALSE;
2919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2928451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
29336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong setShader(JNIEnv* env, jobject clazz, jlong objHandle, jlong shaderHandle) {
2946ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
29536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkShader* shader = reinterpret_cast<SkShader*>(shaderHandle);
29636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return reinterpret_cast<jlong>(obj->setShader(shader));
2979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
2988451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
29936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong setColorFilter(JNIEnv* env, jobject clazz, jlong objHandle, jlong filterHandle) {
3006ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint *>(objHandle);
30136bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkColorFilter* filter  = reinterpret_cast<SkColorFilter *>(filterHandle);
30236bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return reinterpret_cast<jlong>(obj->setColorFilter(filter));
3039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3048451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
30536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong setXfermode(JNIEnv* env, jobject clazz, jlong objHandle, jlong xfermodeHandle) {
3066ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
30736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkXfermode* xfermode = reinterpret_cast<SkXfermode*>(xfermodeHandle);
30836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return reinterpret_cast<jlong>(obj->setXfermode(xfermode));
3099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3108451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
31136bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong setPathEffect(JNIEnv* env, jobject clazz, jlong objHandle, jlong effectHandle) {
3126ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
31336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkPathEffect* effect  = reinterpret_cast<SkPathEffect*>(effectHandle);
31436bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return reinterpret_cast<jlong>(obj->setPathEffect(effect));
3159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3168451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
31736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong setMaskFilter(JNIEnv* env, jobject clazz, jlong objHandle, jlong maskfilterHandle) {
3186ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
31936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkMaskFilter* maskfilter  = reinterpret_cast<SkMaskFilter*>(maskfilterHandle);
32036bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return reinterpret_cast<jlong>(obj->setMaskFilter(maskfilter));
3219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3228451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
32336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong setTypeface(JNIEnv* env, jobject clazz, jlong objHandle, jlong typefaceHandle) {
3246ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        // TODO: in Paint refactoring, set typeface on android Paint, not Paint
325a033630e805c407080221e20b236b6054f324670Raph Levien        return NULL;
3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3278451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
32836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jlong setRasterizer(JNIEnv* env, jobject clazz, jlong objHandle, jlong rasterizerHandle) {
3296ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
3304c9355c35a0f62805739823a62ad41ca1cbc6502Mike Reed        SkAutoTUnref<SkRasterizer> rasterizer(GraphicsJNI::refNativeRasterizer(rasterizerHandle));
33136bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return reinterpret_cast<jlong>(obj->setRasterizer(rasterizer));
3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3338451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
33436bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jint getTextAlign(JNIEnv* env, jobject clazz, jlong objHandle) {
3356ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
33636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        return static_cast<jint>(obj->getTextAlign());
3379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3388451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
33936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static void setTextAlign(JNIEnv* env, jobject clazz, jlong objHandle, jint alignHandle) {
3406ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
3416ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::Align align = static_cast<Paint::Align>(alignHandle);
3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        obj->setTextAlign(align);
3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
3448451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
345ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett    // generate bcp47 identifier for the supplied locale
346ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett    static void toLanguageTag(char* output, size_t outSize,
347ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            const char* locale) {
348ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        if (output == NULL || outSize <= 0) {
349ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            return;
350ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        }
351ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        if (locale == NULL) {
352ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            output[0] = '\0';
353ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            return;
354ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        }
355ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        char canonicalChars[ULOC_FULLNAME_CAPACITY];
356ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        UErrorCode uErr = U_ZERO_ERROR;
357ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        uloc_canonicalize(locale, canonicalChars, ULOC_FULLNAME_CAPACITY,
358ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                &uErr);
359ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        if (U_SUCCESS(uErr)) {
360ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            char likelyChars[ULOC_FULLNAME_CAPACITY];
361ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            uErr = U_ZERO_ERROR;
362ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            uloc_addLikelySubtags(canonicalChars, likelyChars,
363ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                    ULOC_FULLNAME_CAPACITY, &uErr);
364ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            if (U_SUCCESS(uErr)) {
365ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                uErr = U_ZERO_ERROR;
366ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                uloc_toLanguageTag(likelyChars, output, outSize, FALSE, &uErr);
367ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                if (U_SUCCESS(uErr)) {
368ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                    return;
369ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                } else {
370ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                    ALOGD("uloc_toLanguageTag(\"%s\") failed: %s", likelyChars,
371ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                            u_errorName(uErr));
372ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                }
373ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            } else {
374ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                ALOGD("uloc_addLikelySubtags(\"%s\") failed: %s",
375ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                        canonicalChars, u_errorName(uErr));
376ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            }
377ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        } else {
378ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett            ALOGD("uloc_canonicalize(\"%s\") failed: %s", locale,
379ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett                    u_errorName(uErr));
380ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        }
381ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        // unable to build a proper language identifier
382ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        output[0] = '\0';
383ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett    }
384ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett
38536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static void setTextLocale(JNIEnv* env, jobject clazz, jlong objHandle, jstring locale) {
3866ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* obj = reinterpret_cast<Paint*>(objHandle);
387ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        ScopedUtfChars localeChars(env, locale);
388ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        char langTag[ULOC_FULLNAME_CAPACITY];
389ac1cbaf2e5575ac75a0160e13089d51a0bb232faBilly Hewlett        toLanguageTag(langTag, ULOC_FULLNAME_CAPACITY, localeChars.c_str());
390d7a80774f0a2175f9ee81a7741f9a171c78e0b8bDerek Sollenberger
3916f03023ece076d33d8b171d5398c8ed70abf3f54Derek Sollenberger        obj->setTextLocale(langTag);
392517825f1a9f14f92908bd7859b91b927c2eec6d9Fabrice Di Meglio    }
393517825f1a9f14f92908bd7859b91b927c2eec6d9Fabrice Di Meglio
39453c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien    static jboolean isElegantTextHeight(JNIEnv* env, jobject paint) {
39553c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien        NPE_CHECK_RETURN_ZERO(env, paint);
396ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        Paint* obj = getNativePaint(env, paint);
3976f03023ece076d33d8b171d5398c8ed70abf3f54Derek Sollenberger        return obj->getFontVariant() == VARIANT_ELEGANT;
39853c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien    }
39953c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien
40053c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien    static void setElegantTextHeight(JNIEnv* env, jobject paint, jboolean aa) {
40153c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien        NPE_CHECK_RETURN_VOID(env, paint);
402ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        Paint* obj = getNativePaint(env, paint);
4036f03023ece076d33d8b171d5398c8ed70abf3f54Derek Sollenberger        obj->setFontVariant(aa ? VARIANT_ELEGANT : VARIANT_DEFAULT);
40453c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien    }
40553c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien
4069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jfloat getTextSize(JNIEnv* env, jobject paint) {
4079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
408ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        return SkScalarToFloat(getNativePaint(env, paint)->getTextSize());
4099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4108451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
4119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setTextSize(JNIEnv* env, jobject paint, jfloat textSize) {
4129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
413ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setTextSize(textSize);
4149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4158451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jfloat getTextScaleX(JNIEnv* env, jobject paint) {
4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
418ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        return SkScalarToFloat(getNativePaint(env, paint)->getTextScaleX());
4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4208451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setTextScaleX(JNIEnv* env, jobject paint, jfloat scaleX) {
4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
423ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setTextScaleX(scaleX);
4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4258451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jfloat getTextSkewX(JNIEnv* env, jobject paint) {
4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
428ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        return SkScalarToFloat(getNativePaint(env, paint)->getTextSkewX());
4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4308451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static void setTextSkewX(JNIEnv* env, jobject paint, jfloat skewX) {
4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_VOID(env, paint);
433ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        getNativePaint(env, paint)->setTextSkewX(skewX);
4349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
4358451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
436fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod    static jfloat getLetterSpacing(JNIEnv* env, jobject clazz, jlong paintHandle) {
437fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
438fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod        return paint->getLetterSpacing();
439fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod    }
440fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod
441fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod    static void setLetterSpacing(JNIEnv* env, jobject clazz, jlong paintHandle, jfloat letterSpacing) {
442fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
443fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod        paint->setLetterSpacing(letterSpacing);
444fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod    }
445fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod
446805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod    static void setFontFeatureSettings(JNIEnv* env, jobject clazz, jlong paintHandle, jstring settings) {
447805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
448e9ad3931fae71c8a8cd000fd52d5df4be79b0895Behdad Esfahbod        if (!settings) {
449805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod            paint->setFontFeatureSettings(std::string());
450e9ad3931fae71c8a8cd000fd52d5df4be79b0895Behdad Esfahbod        } else {
451805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod            ScopedUtfChars settingsChars(env, settings);
452805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod            paint->setFontFeatureSettings(std::string(settingsChars.c_str(), settingsChars.size()));
453805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod        }
454805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod    }
455805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod
4566ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    static SkScalar getMetricsInternal(JNIEnv* env, jobject jpaint, Paint::FontMetrics *metrics) {
45753c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien        const int kElegantTop = 2500;
45853c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien        const int kElegantBottom = -1000;
459817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        const int kElegantAscent = 1900;
460817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        const int kElegantDescent = -500;
46153c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien        const int kElegantLeading = 0;
462ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        Paint* paint = getNativePaint(env, jpaint);
463ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        TypefaceImpl* typeface = getNativeTypeface(env, jpaint);
464817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        typeface = TypefaceImpl_resolveDefault(typeface);
4651fc0fa87d42ce9268ece76b85b9edc834593e53aRaph Levien        FakedFont baseFont = typeface->fFontCollection->baseFontFaked(typeface->fStyle);
4660627ec08dcc2e87c17e9b85f045dfb27dc4884b3Raph Levien        float saveSkewX = paint->getTextSkewX();
4670627ec08dcc2e87c17e9b85f045dfb27dc4884b3Raph Levien        bool savefakeBold = paint->isFakeBoldText();
4681fc0fa87d42ce9268ece76b85b9edc834593e53aRaph Levien        MinikinFontSkia::populateSkPaint(paint, baseFont.font, baseFont.fakery);
46953c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien        SkScalar spacing = paint->getFontMetrics(metrics);
4700627ec08dcc2e87c17e9b85f045dfb27dc4884b3Raph Levien        // The populateSkPaint call may have changed fake bold / text skew
4710627ec08dcc2e87c17e9b85f045dfb27dc4884b3Raph Levien        // because we want to measure with those effects applied, so now
4720627ec08dcc2e87c17e9b85f045dfb27dc4884b3Raph Levien        // restore the original settings.
4730627ec08dcc2e87c17e9b85f045dfb27dc4884b3Raph Levien        paint->setTextSkewX(saveSkewX);
4740627ec08dcc2e87c17e9b85f045dfb27dc4884b3Raph Levien        paint->setFakeBoldText(savefakeBold);
4756f03023ece076d33d8b171d5398c8ed70abf3f54Derek Sollenberger        if (paint->getFontVariant() == VARIANT_ELEGANT) {
47653c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien            SkScalar size = paint->getTextSize();
47753c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien            metrics->fTop = -size * kElegantTop / 2048;
47853c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien            metrics->fBottom = -size * kElegantBottom / 2048;
47953c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien            metrics->fAscent = -size * kElegantAscent / 2048;
48053c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien            metrics->fDescent = -size * kElegantDescent / 2048;
48153c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien            metrics->fLeading = size * kElegantLeading / 2048;
48253c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien            spacing = metrics->fDescent - metrics->fAscent + metrics->fLeading;
48353c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien        }
48453c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien        return spacing;
48553c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien    }
48653c0077256afebb0312f01ef4f60a7445da1f5d9Raph Levien
487817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien    static jfloat ascent(JNIEnv* env, jobject paint) {
488817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        NPE_CHECK_RETURN_ZERO(env, paint);
4896ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::FontMetrics metrics;
490817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        getMetricsInternal(env, paint, &metrics);
491817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        return SkScalarToFloat(metrics.fAscent);
492817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien    }
493817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien
494817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien    static jfloat descent(JNIEnv* env, jobject paint) {
495817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        NPE_CHECK_RETURN_ZERO(env, paint);
4966ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::FontMetrics metrics;
497817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        getMetricsInternal(env, paint, &metrics);
498817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        return SkScalarToFloat(metrics.fDescent);
499817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien    }
500817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien
5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jfloat getFontMetrics(JNIEnv* env, jobject paint, jobject metricsObj) {
5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
5036ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::FontMetrics metrics;
504817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        SkScalar spacing = getMetricsInternal(env, paint, &metrics);
5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (metricsObj) {
5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            SkASSERT(env->IsInstanceOf(metricsObj, gFontMetrics_class));
5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            env->SetFloatField(metricsObj, gFontMetrics_fieldID.top, SkScalarToFloat(metrics.fTop));
5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            env->SetFloatField(metricsObj, gFontMetrics_fieldID.ascent, SkScalarToFloat(metrics.fAscent));
5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            env->SetFloatField(metricsObj, gFontMetrics_fieldID.descent, SkScalarToFloat(metrics.fDescent));
5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            env->SetFloatField(metricsObj, gFontMetrics_fieldID.bottom, SkScalarToFloat(metrics.fBottom));
5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            env->SetFloatField(metricsObj, gFontMetrics_fieldID.leading, SkScalarToFloat(metrics.fLeading));
5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return SkScalarToFloat(spacing);
5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5168451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    static jint getFontMetricsInt(JNIEnv* env, jobject paint, jobject metricsObj) {
5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, paint);
5196ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::FontMetrics metrics;
5208451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
521817ef4ae79bd39b5e55fc6ca4e3b8e47e114579fRaph Levien        getMetricsInternal(env, paint, &metrics);
52246cb9bdbf56b27cbf2ab878b41d21e30896a4feaLeon Scroggins        int ascent = SkScalarRoundToInt(metrics.fAscent);
52346cb9bdbf56b27cbf2ab878b41d21e30896a4feaLeon Scroggins        int descent = SkScalarRoundToInt(metrics.fDescent);
52446cb9bdbf56b27cbf2ab878b41d21e30896a4feaLeon Scroggins        int leading = SkScalarRoundToInt(metrics.fLeading);
5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (metricsObj) {
5279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            SkASSERT(env->IsInstanceOf(metricsObj, gFontMetricsInt_class));
52846cb9bdbf56b27cbf2ab878b41d21e30896a4feaLeon Scroggins            env->SetIntField(metricsObj, gFontMetricsInt_fieldID.top, SkScalarFloorToInt(metrics.fTop));
5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            env->SetIntField(metricsObj, gFontMetricsInt_fieldID.ascent, ascent);
5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            env->SetIntField(metricsObj, gFontMetricsInt_fieldID.descent, descent);
53146cb9bdbf56b27cbf2ab878b41d21e30896a4feaLeon Scroggins            env->SetIntField(metricsObj, gFontMetricsInt_fieldID.bottom, SkScalarCeilToInt(metrics.fBottom));
5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            env->SetIntField(metricsObj, gFontMetricsInt_fieldID.leading, leading);
5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return descent - ascent + leading;
5359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
53736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jfloat measureText_CIII(JNIEnv* env, jobject jpaint, jcharArray text, jint index, jint count,
538626d3c228116103e95d4429b06dad70f9812561aVictoria Lease            jint bidiFlags) {
5399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, jpaint);
5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, text);
5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        size_t textLength = env->GetArrayLength(text);
5439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if ((index | count) < 0 || (size_t)(index + count) > textLength) {
5448451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes            doThrowAIOOBE(env);
5459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return 0;
5469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
5476ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if (count == 0) {
5486ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
5496ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
5509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
551ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        Paint* paint = getNativePaint(env, jpaint);
5529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* textArray = env->GetCharArrayElements(text, NULL);
5534f810c8535055bd9a8d89a7d1ba0a7c712a8843dFabrice Di Meglio        jfloat result = 0;
554b02d0ca5553300063e4332192632312600caf4b9Fabrice Di Meglio
5551a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        Layout layout;
556ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        TypefaceImpl* typeface = getNativeTypeface(env, jpaint);
55763c5c78a72a21d57913e8601cc2a1ab72a424a02Behdad Esfahbod        MinikinUtils::doLayout(&layout, paint, bidiFlags, typeface, textArray, index, count, textLength);
5581a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        result = layout.getAdvance();
5594f810c8535055bd9a8d89a7d1ba0a7c712a8843dFabrice Di Meglio        env->ReleaseCharArrayElements(text, const_cast<jchar*>(textArray), JNI_ABORT);
5604f810c8535055bd9a8d89a7d1ba0a7c712a8843dFabrice Di Meglio        return result;
5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5628451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
56336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jfloat measureText_StringIII(JNIEnv* env, jobject jpaint, jstring text, jint start, jint end,
564626d3c228116103e95d4429b06dad70f9812561aVictoria Lease            jint bidiFlags) {
5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, jpaint);
5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, text);
5678451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
56851f383d65f9ee3c7d73d0508b576550e7998c5b5Fabrice Di Meglio        size_t textLength = env->GetStringLength(text);
5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int count = end - start;
57051f383d65f9ee3c7d73d0508b576550e7998c5b5Fabrice Di Meglio        if ((start | count) < 0 || (size_t)end > textLength) {
5716ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            doThrowAIOOBE(env);
5726ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
5736ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
5746ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if (count == 0) {
5756ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
5766ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
5776ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio
5786ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        const jchar* textArray = env->GetStringChars(text, NULL);
579ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        Paint* paint = getNativePaint(env, jpaint);
5804f810c8535055bd9a8d89a7d1ba0a7c712a8843dFabrice Di Meglio        jfloat width = 0;
5818451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
5821a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        Layout layout;
583ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        TypefaceImpl* typeface = getNativeTypeface(env, jpaint);
58463c5c78a72a21d57913e8601cc2a1ab72a424a02Behdad Esfahbod        MinikinUtils::doLayout(&layout, paint, bidiFlags, typeface, textArray, start, count, textLength);
5851a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        width = layout.getAdvance();
5864f810c8535055bd9a8d89a7d1ba0a7c712a8843dFabrice Di Meglio
5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseStringChars(text, textArray);
5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return width;
5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
5908451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
591626d3c228116103e95d4429b06dad70f9812561aVictoria Lease    static jfloat measureText_StringI(JNIEnv* env, jobject jpaint, jstring text, jint bidiFlags) {
5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, jpaint);
5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, text);
5948451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        size_t textLength = env->GetStringLength(text);
5966ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if (textLength == 0) {
5976ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
5986ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
5996ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio
6006ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        const jchar* textArray = env->GetStringChars(text, NULL);
601ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        Paint* paint = getNativePaint(env, jpaint);
6024f810c8535055bd9a8d89a7d1ba0a7c712a8843dFabrice Di Meglio        jfloat width = 0;
6036ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio
6041a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        Layout layout;
605ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger        TypefaceImpl* typeface = getNativeTypeface(env, jpaint);
60663c5c78a72a21d57913e8601cc2a1ab72a424a02Behdad Esfahbod        MinikinUtils::doLayout(&layout, paint, bidiFlags, typeface, textArray, 0, textLength, textLength);
6071a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        width = layout.getAdvance();
608b02d0ca5553300063e4332192632312600caf4b9Fabrice Di Meglio
6099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseStringChars(text, textArray);
6109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return width;
6119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6128451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
6136ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    static int dotextwidths(JNIEnv* env, Paint* paint, TypefaceImpl* typeface, const jchar text[], int count,
6141a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien            jfloatArray widths, jint bidiFlags) {
6156ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        NPE_CHECK_RETURN_ZERO(env, paint);
6166ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        NPE_CHECK_RETURN_ZERO(env, text);
6176ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio
6186ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if (count < 0 || !widths) {
6196ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            doThrowAIOOBE(env);
6206ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
6216ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
6226ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if (count == 0) {
6236ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
6246ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
6256ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        size_t widthsLength = env->GetArrayLength(widths);
6266ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if ((size_t)count > widthsLength) {
6276ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            doThrowAIOOBE(env);
6286ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
6296ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
6306ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio
6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        AutoJavaFloatArray autoWidths(env, widths, count);
6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        jfloat* widthsArray = autoWidths.ptr();
6336ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio
6341a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        Layout layout;
63563c5c78a72a21d57913e8601cc2a1ab72a424a02Behdad Esfahbod        MinikinUtils::doLayout(&layout, paint, bidiFlags, typeface, text, 0, count, count);
6361a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        layout.getAdvances(widthsArray);
6379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return count;
6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6408451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
6411a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien    static jint getTextWidths___CIII_F(JNIEnv* env, jobject clazz, jlong paintHandle, jlong typefaceHandle, jcharArray text,
64236bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat            jint index, jint count, jint bidiFlags, jfloatArray widths) {
6436ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
6441a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* textArray = env->GetCharArrayElements(text, NULL);
6461a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        count = dotextwidths(env, paint, typeface, textArray + index, count, widths, bidiFlags);
6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseCharArrayElements(text, const_cast<jchar*>(textArray),
6488451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes                                      JNI_ABORT);
6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return count;
6509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6518451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
6521a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien    static jint getTextWidths__StringIII_F(JNIEnv* env, jobject clazz, jlong paintHandle, jlong typefaceHandle, jstring text,
65336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat            jint start, jint end, jint bidiFlags, jfloatArray widths) {
6546ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
6551a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
6569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* textArray = env->GetStringChars(text, NULL);
6571a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        int count = dotextwidths(env, paint, typeface, textArray + start, end - start, widths, bidiFlags);
6589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseStringChars(text, textArray);
6599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return count;
6609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
6610c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
6626ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    static jfloat doTextRunAdvances(JNIEnv *env, Paint *paint, TypefaceImpl* typeface, const jchar *text,
663051910b9f998030dacb8a0722588cc715813fde1Raph Levien                                    jint start, jint count, jint contextCount, jboolean isRtl,
6640c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt                                    jfloatArray advances, jint advancesIndex) {
6656ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        NPE_CHECK_RETURN_ZERO(env, paint);
6666ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        NPE_CHECK_RETURN_ZERO(env, text);
6676ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio
6686ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if ((start | count | contextCount | advancesIndex) < 0 || contextCount < count) {
6696ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            doThrowAIOOBE(env);
6706ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
6716ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
6726ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if (count == 0) {
6736ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            return 0;
6746ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
6756ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        if (advances) {
6766ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            size_t advancesLength = env->GetArrayLength(advances);
6776ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            if ((size_t)count > advancesLength) {
6786ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio                doThrowAIOOBE(env);
6796ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio                return 0;
6806ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio            }
6816ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        }
6823af2a37e057b31617b8f4eeb22ac3ef85c3fd5aaVictoria Lease        jfloat* advancesArray = new jfloat[count];
6836ab90ed017fb52aac4493a2fac897299d345874fFabrice Di Meglio        jfloat totalAdvance = 0;
6840c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
685051910b9f998030dacb8a0722588cc715813fde1Raph Levien        int bidiFlags = isRtl ? kBidi_Force_RTL : kBidi_Force_LTR;
686051910b9f998030dacb8a0722588cc715813fde1Raph Levien
6871a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        Layout layout;
68863c5c78a72a21d57913e8601cc2a1ab72a424a02Behdad Esfahbod        MinikinUtils::doLayout(&layout, paint, bidiFlags, typeface, text, start, count, contextCount);
6891a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        layout.getAdvances(advancesArray);
6901a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        totalAdvance = layout.getAdvance();
6910c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
6920c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        if (advances != NULL) {
6930c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt            env->SetFloatArrayRegion(advances, advancesIndex, count, advancesArray);
6940c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        }
6953af2a37e057b31617b8f4eeb22ac3ef85c3fd5aaVictoria Lease        delete [] advancesArray;
6960c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        return totalAdvance;
6970c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt    }
6980c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
699051910b9f998030dacb8a0722588cc715813fde1Raph Levien    static jfloat getTextRunAdvances___CIIIIZ_FI(JNIEnv* env, jobject clazz, jlong paintHandle,
7001a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien            jlong typefaceHandle,
7010c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt            jcharArray text, jint index, jint count, jint contextIndex, jint contextCount,
702051910b9f998030dacb8a0722588cc715813fde1Raph Levien            jboolean isRtl, jfloatArray advances, jint advancesIndex) {
7036ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
7041a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
7050c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        jchar* textArray = env->GetCharArrayElements(text, NULL);
7061a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        jfloat result = doTextRunAdvances(env, paint, typeface, textArray + contextIndex,
707051910b9f998030dacb8a0722588cc715813fde1Raph Levien                index - contextIndex, count, contextCount, isRtl, advances, advancesIndex);
7080c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        env->ReleaseCharArrayElements(text, textArray, JNI_ABORT);
7090c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        return result;
7100c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt    }
7110c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
712051910b9f998030dacb8a0722588cc715813fde1Raph Levien    static jfloat getTextRunAdvances__StringIIIIZ_FI(JNIEnv* env, jobject clazz, jlong paintHandle,
7131a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien            jlong typefaceHandle,
714051910b9f998030dacb8a0722588cc715813fde1Raph Levien            jstring text, jint start, jint end, jint contextStart, jint contextEnd, jboolean isRtl,
715665f02c66702b77db25e950ff433230186bc3243Fabrice Di Meglio            jfloatArray advances, jint advancesIndex) {
7166ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
7171a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
7180c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        const jchar* textArray = env->GetStringChars(text, NULL);
7191a73f732f91e97c9c66b808c245ddda36a10e987Raph Levien        jfloat result = doTextRunAdvances(env, paint, typeface, textArray + contextStart,
720051910b9f998030dacb8a0722588cc715813fde1Raph Levien                start - contextStart, end - start, contextEnd - contextStart, isRtl,
721665f02c66702b77db25e950ff433230186bc3243Fabrice Di Meglio                advances, advancesIndex);
722eee49c699c035ffba188417489f40d34f587d65cFabrice Di Meglio        env->ReleaseStringChars(text, textArray);
723eee49c699c035ffba188417489f40d34f587d65cFabrice Di Meglio        return result;
724eee49c699c035ffba188417489f40d34f587d65cFabrice Di Meglio    }
725eee49c699c035ffba188417489f40d34f587d65cFabrice Di Meglio
7266ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    static jint doTextRunCursor(JNIEnv *env, Paint* paint, const jchar *text, jint start,
727a4bf8114a73371b51a0eba9d5d61dfeb597a1abbFabrice Di Meglio            jint count, jint flags, jint offset, jint opt) {
728e368b6bcfd499543a876b4cae02f1ca563ff1cb6Raph Levien        GraphemeBreak::MoveOpt moveOpt = GraphemeBreak::MoveOpt(opt);
729e368b6bcfd499543a876b4cae02f1ca563ff1cb6Raph Levien        size_t result = GraphemeBreak::getTextRunCursor(text, start, count, offset, moveOpt);
730e368b6bcfd499543a876b4cae02f1ca563ff1cb6Raph Levien        return static_cast<jint>(result);
7310c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt    }
7320c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
73336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jint getTextRunCursor___C(JNIEnv* env, jobject clazz, jlong paintHandle, jcharArray text,
734051910b9f998030dacb8a0722588cc715813fde1Raph Levien            jint contextStart, jint contextCount, jint dir, jint offset, jint cursorOpt) {
7356ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
7360c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        jchar* textArray = env->GetCharArrayElements(text, NULL);
737051910b9f998030dacb8a0722588cc715813fde1Raph Levien        jint result = doTextRunCursor(env, paint, textArray, contextStart, contextCount, dir,
7380c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt                offset, cursorOpt);
7390c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        env->ReleaseCharArrayElements(text, textArray, JNI_ABORT);
7400c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        return result;
7410c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt    }
7420c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
74336bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    static jint getTextRunCursor__String(JNIEnv* env, jobject clazz, jlong paintHandle, jstring text,
744051910b9f998030dacb8a0722588cc715813fde1Raph Levien            jint contextStart, jint contextEnd, jint dir, jint offset, jint cursorOpt) {
7456ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
7460c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        const jchar* textArray = env->GetStringChars(text, NULL);
7470c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        jint result = doTextRunCursor(env, paint, textArray, contextStart,
748051910b9f998030dacb8a0722588cc715813fde1Raph Levien                contextEnd - contextStart, dir, offset, cursorOpt);
7490c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        env->ReleaseStringChars(text, textArray);
7500c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt        return result;
7510c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt    }
7520c702b88c5d0d4380930b920f5be6e66dd95a0d8Doug Felt
753f2114d5646194100242206b267ddd6e7194b7da9Raph Levien    class GetTextFunctor {
754f2114d5646194100242206b267ddd6e7194b7da9Raph Levien    public:
7556ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        GetTextFunctor(const Layout& layout, SkPath* path, jfloat x, jfloat y, Paint* paint,
756f2114d5646194100242206b267ddd6e7194b7da9Raph Levien                    uint16_t* glyphs, SkPoint* pos)
757f2114d5646194100242206b267ddd6e7194b7da9Raph Levien                : layout(layout), path(path), x(x), y(y), paint(paint), glyphs(glyphs), pos(pos) {
758f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        }
759f2114d5646194100242206b267ddd6e7194b7da9Raph Levien
7601fc0fa87d42ce9268ece76b85b9edc834593e53aRaph Levien        void operator()(size_t start, size_t end) {
761f2114d5646194100242206b267ddd6e7194b7da9Raph Levien            for (size_t i = start; i < end; i++) {
762f2114d5646194100242206b267ddd6e7194b7da9Raph Levien                glyphs[i] = layout.getGlyphId(i);
763f2114d5646194100242206b267ddd6e7194b7da9Raph Levien                pos[i].fX = x + layout.getX(i);
764f2114d5646194100242206b267ddd6e7194b7da9Raph Levien                pos[i].fY = y + layout.getY(i);
765f2114d5646194100242206b267ddd6e7194b7da9Raph Levien            }
766f2114d5646194100242206b267ddd6e7194b7da9Raph Levien            if (start == 0) {
767f2114d5646194100242206b267ddd6e7194b7da9Raph Levien                paint->getPosTextPath(glyphs + start, (end - start) << 1, pos + start, path);
768f2114d5646194100242206b267ddd6e7194b7da9Raph Levien            } else {
769f2114d5646194100242206b267ddd6e7194b7da9Raph Levien                paint->getPosTextPath(glyphs + start, (end - start) << 1, pos + start, &tmpPath);
770f2114d5646194100242206b267ddd6e7194b7da9Raph Levien                path->addPath(tmpPath);
771f2114d5646194100242206b267ddd6e7194b7da9Raph Levien            }
772f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        }
773f2114d5646194100242206b267ddd6e7194b7da9Raph Levien    private:
774f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        const Layout& layout;
775f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        SkPath* path;
776f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        jfloat x;
777f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        jfloat y;
7786ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint;
779f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        uint16_t* glyphs;
780f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        SkPoint* pos;
781f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        SkPath tmpPath;
782f2114d5646194100242206b267ddd6e7194b7da9Raph Levien    };
783f2114d5646194100242206b267ddd6e7194b7da9Raph Levien
7846ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    static void getTextPath(JNIEnv* env, Paint* paint, TypefaceImpl* typeface, const jchar* text,
785f2114d5646194100242206b267ddd6e7194b7da9Raph Levien            jint count, jint bidiFlags, jfloat x, jfloat y, SkPath* path) {
786f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        Layout layout;
78763c5c78a72a21d57913e8601cc2a1ab72a424a02Behdad Esfahbod        MinikinUtils::doLayout(&layout, paint, bidiFlags, typeface, text, 0, count, count);
788f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        size_t nGlyphs = layout.nGlyphs();
789f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        uint16_t* glyphs = new uint16_t[nGlyphs];
790f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        SkPoint* pos = new SkPoint[nGlyphs];
791f2114d5646194100242206b267ddd6e7194b7da9Raph Levien
792f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        x += MinikinUtils::xOffsetForTextAlign(paint, layout);
7936ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint::Align align = paint->getTextAlign();
7946ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        paint->setTextAlign(Paint::kLeft_Align);
7956ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        paint->setTextEncoding(Paint::kGlyphID_TextEncoding);
796f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        GetTextFunctor f(layout, path, x, y, paint, glyphs, pos);
7971fc0fa87d42ce9268ece76b85b9edc834593e53aRaph Levien        MinikinUtils::forFontRun(layout, paint, f);
798f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        paint->setTextAlign(align);
799f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        delete[] glyphs;
800f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        delete[] pos;
801f7cb1f75fdaedf996cab7c4690b080adc7bc5b97Doug Felt    }
802f7cb1f75fdaedf996cab7c4690b080adc7bc5b97Doug Felt
803f2114d5646194100242206b267ddd6e7194b7da9Raph Levien    static void getTextPath___C(JNIEnv* env, jobject clazz, jlong paintHandle,
804f2114d5646194100242206b267ddd6e7194b7da9Raph Levien            jlong typefaceHandle, jint bidiFlags,
80536bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat            jcharArray text, jint index, jint count, jfloat x, jfloat y, jlong pathHandle) {
8066ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
807f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
80836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkPath* path = reinterpret_cast<SkPath*>(pathHandle);
8099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* textArray = env->GetCharArrayElements(text, NULL);
810f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        getTextPath(env, paint, typeface, textArray + index, count, bidiFlags, x, y, path);
811f7cb1f75fdaedf996cab7c4690b080adc7bc5b97Doug Felt        env->ReleaseCharArrayElements(text, const_cast<jchar*>(textArray), JNI_ABORT);
8129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
813f7cb1f75fdaedf996cab7c4690b080adc7bc5b97Doug Felt
814f2114d5646194100242206b267ddd6e7194b7da9Raph Levien    static void getTextPath__String(JNIEnv* env, jobject clazz, jlong paintHandle,
815f2114d5646194100242206b267ddd6e7194b7da9Raph Levien            jlong typefaceHandle, jint bidiFlags,
81636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat            jstring text, jint start, jint end, jfloat x, jfloat y, jlong pathHandle) {
8176ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
818f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
81936bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat        SkPath* path = reinterpret_cast<SkPath*>(pathHandle);
8209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* textArray = env->GetStringChars(text, NULL);
821f2114d5646194100242206b267ddd6e7194b7da9Raph Levien        getTextPath(env, paint, typeface, textArray + start, end - start, bidiFlags, x, y, path);
8229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseStringChars(text, textArray);
8239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
824f7cb1f75fdaedf996cab7c4690b080adc7bc5b97Doug Felt
825c29a0a4664a4b9871fadd668b632469a0db240b9Derek Sollenberger    static void setShadowLayer(JNIEnv* env, jobject clazz, jlong paintHandle, jfloat radius,
82636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat                               jfloat dx, jfloat dy, jint color) {
8276ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
8289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (radius <= 0) {
8299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            paint->setLooper(NULL);
8309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        else {
832c29a0a4664a4b9871fadd668b632469a0db240b9Derek Sollenberger            SkScalar sigma = android::uirenderer::Blur::convertRadiusToSigma(radius);
83349f0db34811bc374eb6af0a6ba233325ff6f14ccVictoria Lease            paint->setLooper(SkBlurDrawLooper::Create((SkColor)color, sigma, dx, dy))->unref();
8349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
837c29a0a4664a4b9871fadd668b632469a0db240b9Derek Sollenberger    static jboolean hasShadowLayer(JNIEnv* env, jobject clazz, jlong paintHandle) {
8386ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
839c29a0a4664a4b9871fadd668b632469a0db240b9Derek Sollenberger        return paint->getLooper() && paint->getLooper()->asABlurShadow(NULL);
840c29a0a4664a4b9871fadd668b632469a0db240b9Derek Sollenberger    }
841c29a0a4664a4b9871fadd668b632469a0db240b9Derek Sollenberger
8426ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    static int breakText(JNIEnv* env, const Paint& paint, TypefaceImpl* typeface, const jchar text[],
843626d3c228116103e95d4429b06dad70f9812561aVictoria Lease                         int count, float maxWidth, jint bidiFlags, jfloatArray jmeasured,
84457c4fed236cbec0944824b534fb9c828ac0d1f81Derek Sollenberger                         const bool forwardScan) {
845a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        size_t measuredCount = 0;
846a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        float measured = 0;
847a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye
848a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        Layout layout;
84963c5c78a72a21d57913e8601cc2a1ab72a424a02Behdad Esfahbod        MinikinUtils::doLayout(&layout, &paint, bidiFlags, typeface, text, 0, count, count);
850a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        float* advances = new float[count];
851a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        layout.getAdvances(advances);
85257c4fed236cbec0944824b534fb9c828ac0d1f81Derek Sollenberger
853a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        for (int i = 0; i < count; i++) {
854a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            // traverse in the given direction
855a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            int index = forwardScan ? i : (count - i - 1);
856a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            float width = advances[index];
857a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            if (measured + width > maxWidth) {
858a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye                break;
859a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            }
860a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            // properly handle clusters when scanning backwards
861a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            if (forwardScan || width != 0.0f) {
862a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye                measuredCount = i + 1;
863a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            }
864a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            measured += width;
865a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        }
866a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        delete[] advances;
8679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (jmeasured && env->GetArrayLength(jmeasured) > 0) {
8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            AutoJavaFloatArray autoMeasured(env, jmeasured, 1);
8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            jfloat* array = autoMeasured.ptr();
871a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye            array[0] = measured;
8729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
873a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        return measuredCount;
8749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
8759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
876a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye    static jint breakTextC(JNIEnv* env, jobject clazz, jlong paintHandle, jlong typefaceHandle, jcharArray jtext,
87736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat            jint index, jint count, jfloat maxWidth, jint bidiFlags, jfloatArray jmeasuredWidth) {
8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, jtext);
8799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8806ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
881a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
882a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye
88357c4fed236cbec0944824b534fb9c828ac0d1f81Derek Sollenberger        bool forwardTextDirection;
8849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if (count < 0) {
88557c4fed236cbec0944824b534fb9c828ac0d1f81Derek Sollenberger            forwardTextDirection = false;
8869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            count = -count;
8879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        else {
88957c4fed236cbec0944824b534fb9c828ac0d1f81Derek Sollenberger            forwardTextDirection = true;
8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        if ((index < 0) || (index + count > env->GetArrayLength(jtext))) {
8938451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes            doThrowAIOOBE(env);
8949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project            return 0;
8959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        }
8969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
8979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* text = env->GetCharArrayElements(jtext, NULL);
898a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        count = breakText(env, *paint, typeface, text + index, count, maxWidth,
89957c4fed236cbec0944824b534fb9c828ac0d1f81Derek Sollenberger                          bidiFlags, jmeasuredWidth, forwardTextDirection);
9009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseCharArrayElements(jtext, const_cast<jchar*>(text),
9019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                      JNI_ABORT);
9029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return count;
9039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
905a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye    static jint breakTextS(JNIEnv* env, jobject clazz, jlong paintHandle, jlong typefaceHandle, jstring jtext,
90636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat                jboolean forwards, jfloat maxWidth, jint bidiFlags, jfloatArray jmeasuredWidth) {
9079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        NPE_CHECK_RETURN_ZERO(env, jtext);
9089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9096ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        Paint* paint = reinterpret_cast<Paint*>(paintHandle);
910a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
911a7aa1b0aa566b2ff310cb89fbc9437de4819f583Anish Athalye
9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        int count = env->GetStringLength(jtext);
9139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* text = env->GetStringChars(jtext, NULL);
91457c4fed236cbec0944824b534fb9c828ac0d1f81Derek Sollenberger        count = breakText(env, *paint, typeface, text, count, maxWidth, bidiFlags, jmeasuredWidth, forwards);
9159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseStringChars(jtext, text);
9169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        return count;
9179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
919854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien    static void doTextBounds(JNIEnv* env, const jchar* text, int count, jobject bounds,
9206ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod            const Paint& paint, TypefaceImpl* typeface, jint bidiFlags) {
921059e12ccd20f5c249724a8362d6bac325334ea76Romain Guy        SkRect  r;
9229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        SkIRect ir;
9238451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
924854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        Layout layout;
92563c5c78a72a21d57913e8601cc2a1ab72a424a02Behdad Esfahbod        MinikinUtils::doLayout(&layout, &paint, bidiFlags, typeface, text, 0, count, count);
926854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        MinikinRect rect;
927854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        layout.getBounds(&rect);
928854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        r.fLeft = rect.mLeft;
929854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        r.fTop = rect.mTop;
930854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        r.fRight = rect.mRight;
931854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        r.fBottom = rect.mBottom;
9329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        r.roundOut(&ir);
9339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        GraphicsJNI::irect_to_jrect(ir, env, bounds);
9349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
936854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien    static void getStringBounds(JNIEnv* env, jobject, jlong paintHandle, jlong typefaceHandle,
93736bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat                                jstring text, jint start, jint end, jint bidiFlags, jobject bounds) {
9386ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        const Paint* paint = reinterpret_cast<Paint*>(paintHandle);;
939854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
9409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* textArray = env->GetStringChars(text, NULL);
941854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        doTextBounds(env, textArray + start, end - start, bounds, *paint, typeface, bidiFlags);
9429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseStringChars(text, textArray);
9439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9448451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
945854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien    static void getCharArrayBounds(JNIEnv* env, jobject, jlong paintHandle, jlong typefaceHandle,
94636bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat                        jcharArray text, jint index, jint count, jint bidiFlags, jobject bounds) {
9476ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        const Paint* paint = reinterpret_cast<Paint*>(paintHandle);
948854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        TypefaceImpl* typeface = reinterpret_cast<TypefaceImpl*>(typefaceHandle);
9499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        const jchar* textArray = env->GetCharArrayElements(text, NULL);
950854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien        doTextBounds(env, textArray + index, count, bounds, *paint, typeface, bidiFlags);
9519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        env->ReleaseCharArrayElements(text, const_cast<jchar*>(textArray),
9529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project                                      JNI_ABORT);
9539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    }
9548451b25a4422656bbd6657a5855e69c0f4d53c74Elliott Hughes
9559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project};
9569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
9579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic JNINativeMethod methods[] = {
9586ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"finalizer", "(J)V", (void*) PaintGlue::finalizer},
9596ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_init","()J", (void*) PaintGlue::init},
9606ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_initWithPaint","(J)J", (void*) PaintGlue::initWithPaint},
9616ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_reset","(J)V", (void*) PaintGlue::reset},
9626ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_set","(JJ)V", (void*) PaintGlue::assign},
9636ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getFlags","()I", (void*) PaintGlue::getFlags},
9646ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setFlags","(I)V", (void*) PaintGlue::setFlags},
9656ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getHinting","()I", (void*) PaintGlue::getHinting},
9666ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setHinting","(I)V", (void*) PaintGlue::setHinting},
9676ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setAntiAlias","(Z)V", (void*) PaintGlue::setAntiAlias},
9686ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setSubpixelText","(Z)V", (void*) PaintGlue::setSubpixelText},
9696ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setLinearText","(Z)V", (void*) PaintGlue::setLinearText},
9706ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setUnderlineText","(Z)V", (void*) PaintGlue::setUnderlineText},
9716ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setStrikeThruText","(Z)V", (void*) PaintGlue::setStrikeThruText},
9726ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setFakeBoldText","(Z)V", (void*) PaintGlue::setFakeBoldText},
9736ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setFilterBitmap","(Z)V", (void*) PaintGlue::setFilterBitmap},
9746ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setDither","(Z)V", (void*) PaintGlue::setDither},
9756ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getStyle","(J)I", (void*) PaintGlue::getStyle},
9766ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setStyle","(JI)V", (void*) PaintGlue::setStyle},
9776ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getColor","()I", (void*) PaintGlue::getColor},
9786ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setColor","(I)V", (void*) PaintGlue::setColor},
9796ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getAlpha","()I", (void*) PaintGlue::getAlpha},
9806ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setAlpha","(I)V", (void*) PaintGlue::setAlpha},
9816ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getStrokeWidth","()F", (void*) PaintGlue::getStrokeWidth},
9826ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setStrokeWidth","(F)V", (void*) PaintGlue::setStrokeWidth},
9836ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getStrokeMiter","()F", (void*) PaintGlue::getStrokeMiter},
9846ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setStrokeMiter","(F)V", (void*) PaintGlue::setStrokeMiter},
9856ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getStrokeCap","(J)I", (void*) PaintGlue::getStrokeCap},
9866ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setStrokeCap","(JI)V", (void*) PaintGlue::setStrokeCap},
9876ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getStrokeJoin","(J)I", (void*) PaintGlue::getStrokeJoin},
9886ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setStrokeJoin","(JI)V", (void*) PaintGlue::setStrokeJoin},
9896ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getFillPath","(JJJ)Z", (void*) PaintGlue::getFillPath},
9906ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setShader","(JJ)J", (void*) PaintGlue::setShader},
9916ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setColorFilter","(JJ)J", (void*) PaintGlue::setColorFilter},
9926ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setXfermode","(JJ)J", (void*) PaintGlue::setXfermode},
9936ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setPathEffect","(JJ)J", (void*) PaintGlue::setPathEffect},
9946ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setMaskFilter","(JJ)J", (void*) PaintGlue::setMaskFilter},
9956ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setTypeface","(JJ)J", (void*) PaintGlue::setTypeface},
9966ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setRasterizer","(JJ)J", (void*) PaintGlue::setRasterizer},
9976ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getTextAlign","(J)I", (void*) PaintGlue::getTextAlign},
9986ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setTextAlign","(JI)V", (void*) PaintGlue::setTextAlign},
9996ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setTextLocale","(JLjava/lang/String;)V", (void*) PaintGlue::setTextLocale},
10006ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"isElegantTextHeight","()Z", (void*) PaintGlue::isElegantTextHeight},
10016ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setElegantTextHeight","(Z)V", (void*) PaintGlue::setElegantTextHeight},
10026ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getTextSize","()F", (void*) PaintGlue::getTextSize},
10036ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setTextSize","(F)V", (void*) PaintGlue::setTextSize},
10046ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getTextScaleX","()F", (void*) PaintGlue::getTextScaleX},
10056ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setTextScaleX","(F)V", (void*) PaintGlue::setTextScaleX},
10066ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getTextSkewX","()F", (void*) PaintGlue::getTextSkewX},
10076ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"setTextSkewX","(F)V", (void*) PaintGlue::setTextSkewX},
1008fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod    {"native_getLetterSpacing","(J)F", (void*) PaintGlue::getLetterSpacing},
1009fa80f7491df82d71b7084500519a2195afbea706Behdad Esfahbod    {"native_setLetterSpacing","(JF)V", (void*) PaintGlue::setLetterSpacing},
1010805f6ebf17e2791624bb1a30834b4c1cc65583bfBehdad Esfahbod    {"native_setFontFeatureSettings","(JLjava/lang/String;)V", (void*) PaintGlue::setFontFeatureSettings},
10116ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"ascent","()F", (void*) PaintGlue::ascent},
10126ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"descent","()F", (void*) PaintGlue::descent},
10136ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getFontMetrics", "(Landroid/graphics/Paint$FontMetrics;)F", (void*)PaintGlue::getFontMetrics},
10146ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"getFontMetricsInt", "(Landroid/graphics/Paint$FontMetricsInt;)I", (void*)PaintGlue::getFontMetricsInt},
10156ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_measureText","([CIII)F", (void*) PaintGlue::measureText_CIII},
10166ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_measureText","(Ljava/lang/String;I)F", (void*) PaintGlue::measureText_StringI},
10176ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_measureText","(Ljava/lang/String;III)F", (void*) PaintGlue::measureText_StringIII},
10186ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_breakText","(JJ[CIIFI[F)I", (void*) PaintGlue::breakTextC},
10196ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_breakText","(JJLjava/lang/String;ZFI[F)I", (void*) PaintGlue::breakTextS},
10206ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getTextWidths","(JJ[CIII[F)I", (void*) PaintGlue::getTextWidths___CIII_F},
10216ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getTextWidths","(JJLjava/lang/String;III[F)I", (void*) PaintGlue::getTextWidths__StringIII_F},
1022051910b9f998030dacb8a0722588cc715813fde1Raph Levien    {"native_getTextRunAdvances","(JJ[CIIIIZ[FI)F",
10236ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        (void*) PaintGlue::getTextRunAdvances___CIIIIZ_FI},
1024051910b9f998030dacb8a0722588cc715813fde1Raph Levien    {"native_getTextRunAdvances","(JJLjava/lang/String;IIIIZ[FI)F",
10256ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        (void*) PaintGlue::getTextRunAdvances__StringIIIIZ_FI},
1026da12f389eb4be0c08ca3fa9ca7663f4977858df5Fabrice Di Meglio
10276ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getTextRunCursor", "(J[CIIIII)I", (void*) PaintGlue::getTextRunCursor___C},
102836bef0bf30d6bae48cf3837df351075ca4fce654Ashok Bhat    {"native_getTextRunCursor", "(JLjava/lang/String;IIIII)I",
10296ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod        (void*) PaintGlue::getTextRunCursor__String},
10306ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getTextPath","(JJI[CIIFFJ)V", (void*) PaintGlue::getTextPath___C},
10316ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_getTextPath","(JJILjava/lang/String;IIFFJ)V", (void*) PaintGlue::getTextPath__String},
1032854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien    {"nativeGetStringBounds", "(JJLjava/lang/String;IIILandroid/graphics/Rect;)V",
10336ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod                                        (void*) PaintGlue::getStringBounds },
1034854363e3d2fb2a9e41d8719f4a2f0f8b89440a46Raph Levien    {"nativeGetCharArrayBounds", "(JJ[CIIILandroid/graphics/Rect;)V",
10356ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod                                    (void*) PaintGlue::getCharArrayBounds },
10366ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_setShadowLayer", "(JFFFI)V", (void*)PaintGlue::setShadowLayer},
10376ba30b85ddfbe37c338ee8dde3dd33322eb38d47Behdad Esfahbod    {"native_hasShadowLayer", "(J)Z", (void*)PaintGlue::hasShadowLayer}
10389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project};
10399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1040ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenbergerstatic jclass makeGlobalRef(JNIEnv* env, const char classname[])
1041ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger{
1042ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    jclass c = env->FindClass(classname);
1043ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    SkASSERT(c);
1044ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    return (jclass) env->NewGlobalRef(c);
1045ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger}
1046ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger
10479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectstatic jfieldID req_fieldID(jfieldID id) {
10489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    SkASSERT(id);
10499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return id;
10509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectint register_android_graphics_Paint(JNIEnv* env) {
1053ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    gFontMetrics_class = makeGlobalRef(env, "android/graphics/Paint$FontMetrics");
10549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetrics_fieldID.top = req_fieldID(env->GetFieldID(gFontMetrics_class, "top", "F"));
10559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetrics_fieldID.ascent = req_fieldID(env->GetFieldID(gFontMetrics_class, "ascent", "F"));
10569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetrics_fieldID.descent = req_fieldID(env->GetFieldID(gFontMetrics_class, "descent", "F"));
10579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetrics_fieldID.bottom = req_fieldID(env->GetFieldID(gFontMetrics_class, "bottom", "F"));
10589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetrics_fieldID.leading = req_fieldID(env->GetFieldID(gFontMetrics_class, "leading", "F"));
10599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1060ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    gFontMetricsInt_class = makeGlobalRef(env, "android/graphics/Paint$FontMetricsInt");
10619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetricsInt_fieldID.top = req_fieldID(env->GetFieldID(gFontMetricsInt_class, "top", "I"));
10629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetricsInt_fieldID.ascent = req_fieldID(env->GetFieldID(gFontMetricsInt_class, "ascent", "I"));
10639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetricsInt_fieldID.descent = req_fieldID(env->GetFieldID(gFontMetricsInt_class, "descent", "I"));
10649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetricsInt_fieldID.bottom = req_fieldID(env->GetFieldID(gFontMetricsInt_class, "bottom", "I"));
10659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    gFontMetricsInt_fieldID.leading = req_fieldID(env->GetFieldID(gFontMetricsInt_class, "leading", "I"));
10669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
1067ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    gPaint_class = makeGlobalRef(env, "android/graphics/Paint");
1068ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    gPaint_nativeInstanceID = req_fieldID(env->GetFieldID(gPaint_class, "mNativePaint", "J"));
1069ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger    gPaint_nativeTypefaceID = req_fieldID(env->GetFieldID(gPaint_class, "mNativeTypeface", "J"));
1070ab22c1c792bc5f422a029a4ab6a23861e44136d8Derek Sollenberger
10719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    int result = AndroidRuntime::registerNativeMethods(env, "android/graphics/Paint", methods,
10729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project        sizeof(methods) / sizeof(methods[0]));
10739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project    return result;
10749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
10759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project
10769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project}
1077