blob: f6ae169ae3f039fb53bbc1f47c6aa8ccc26e7f3e [file] [log] [blame]
Seigo Nonaka6f1868b2017-12-01 16:24:19 -08001/*
2 * Copyright (C) 2017 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Seigo Nonakaa5534772018-03-15 00:22:20 -070017#include "GraphicsJNI.h"
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080018#include "utils/misc.h"
19#include "utils/Log.h"
20#include <nativehelper/ScopedStringChars.h>
21#include <nativehelper/ScopedPrimitiveArray.h>
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080022#include <cstdint>
23#include <vector>
24#include <list>
25#include <algorithm>
26
27#include "SkPaint.h"
28#include "SkTypeface.h"
29#include <hwui/MinikinSkia.h>
30#include <hwui/MinikinUtils.h>
31#include <hwui/Paint.h>
32#include <minikin/FontCollection.h>
33#include <minikin/AndroidLineBreakerHelper.h>
34#include <minikin/MinikinFont.h>
35
36namespace android {
37
38static inline minikin::MeasuredTextBuilder* toBuilder(jlong ptr) {
39 return reinterpret_cast<minikin::MeasuredTextBuilder*>(ptr);
40}
41
42static inline Paint* toPaint(jlong ptr) {
43 return reinterpret_cast<Paint*>(ptr);
44}
45
Seigo Nonaka9d3bd082018-01-11 10:02:12 -080046static inline minikin::MeasuredText* toMeasuredParagraph(jlong ptr) {
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080047 return reinterpret_cast<minikin::MeasuredText*>(ptr);
48}
49
50template<typename Ptr> static inline jlong toJLong(Ptr ptr) {
51 return reinterpret_cast<jlong>(ptr);
52}
53
Seigo Nonaka9d3bd082018-01-11 10:02:12 -080054static void releaseMeasuredParagraph(jlong measuredTextPtr) {
55 delete toMeasuredParagraph(measuredTextPtr);
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080056}
57
58// Regular JNI
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +010059static jlong nInitBuilder(CRITICAL_JNI_PARAMS) {
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080060 return toJLong(new minikin::MeasuredTextBuilder());
61}
62
63// Regular JNI
64static void nAddStyleRun(JNIEnv* /* unused */, jclass /* unused */, jlong builderPtr,
James.cf Lin664a75b2022-01-19 16:56:34 +080065 jlong paintPtr, jint lbStyle, jint lbWordStyle, jint start, jint end,
66 jboolean isRtl) {
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080067 Paint* paint = toPaint(paintPtr);
68 const Typeface* typeface = Typeface::resolveDefault(paint->getAndroidTypeface());
69 minikin::MinikinPaint minikinPaint = MinikinUtils::prepareMinikinPaint(paint, typeface);
James.cf Lin664a75b2022-01-19 16:56:34 +080070 toBuilder(builderPtr)
71 ->addStyleRun(start, end, std::move(minikinPaint), lbStyle, lbWordStyle, isRtl);
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080072}
73
74// Regular JNI
75static void nAddReplacementRun(JNIEnv* /* unused */, jclass /* unused */, jlong builderPtr,
76 jlong paintPtr, jint start, jint end, jfloat width) {
77 toBuilder(builderPtr)->addReplacementRun(start, end, width,
78 toPaint(paintPtr)->getMinikinLocaleListId());
79}
80
81// Regular JNI
Seigo Nonaka0f761402021-08-30 01:27:10 +000082static jlong nBuildMeasuredText(JNIEnv* env, jclass /* unused */, jlong builderPtr, jlong hintPtr,
83 jcharArray javaText, jboolean computeHyphenation,
Seigo Nonaka6e39daf2023-08-19 11:36:26 +090084 jboolean computeLayout, jboolean computeBounds,
85 jboolean fastHyphenationMode) {
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080086 ScopedCharArrayRO text(env, javaText);
87 const minikin::U16StringPiece textBuffer(text.get(), text.size());
88
89 // Pass the ownership to Java.
Seigo Nonaka0f761402021-08-30 01:27:10 +000090 return toJLong(toBuilder(builderPtr)
Seigo Nonaka6e39daf2023-08-19 11:36:26 +090091 ->build(textBuffer, computeHyphenation, computeLayout, computeBounds,
Seigo Nonaka0f761402021-08-30 01:27:10 +000092 fastHyphenationMode, toMeasuredParagraph(hintPtr))
93 .release());
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080094}
95
96// Regular JNI
97static void nFreeBuilder(JNIEnv* env, jclass /* unused */, jlong builderPtr) {
98 delete toBuilder(builderPtr);
99}
100
101// CriticalNative
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +0100102static jfloat nGetWidth(CRITICAL_JNI_PARAMS_COMMA jlong ptr, jint start, jint end) {
Seigo Nonaka783f9612018-01-20 12:11:13 -0800103 minikin::MeasuredText* mt = toMeasuredParagraph(ptr);
104 float r = 0.0f;
105 for (int i = start; i < end; ++i) {
106 r += mt->widths[i];
107 }
108 return r;
109}
110
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +0100111static jfloat nGetCharWidthAt(CRITICAL_JNI_PARAMS_COMMA jlong ptr, jint offset) {
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700112 return toMeasuredParagraph(ptr)->widths[offset];
113}
114
Seigo Nonakaa5534772018-03-15 00:22:20 -0700115// Regular JNI
Seigo Nonakafb0abe12018-04-02 23:25:38 -0700116static void nGetBounds(JNIEnv* env, jobject, jlong ptr, jcharArray javaText, jint start, jint end,
117 jobject bounds) {
Seigo Nonakaa5534772018-03-15 00:22:20 -0700118 ScopedCharArrayRO text(env, javaText);
119 const minikin::U16StringPiece textBuffer(text.get(), text.size());
Seigo Nonakafb0abe12018-04-02 23:25:38 -0700120 const minikin::Range range(start, end);
Seigo Nonakaa5534772018-03-15 00:22:20 -0700121
Seigo Nonakafb0abe12018-04-02 23:25:38 -0700122 minikin::MinikinRect rect = toMeasuredParagraph(ptr)->getBounds(textBuffer, range);
Seigo Nonakaa5534772018-03-15 00:22:20 -0700123
124 SkRect r;
125 r.fLeft = rect.mLeft;
126 r.fTop = rect.mTop;
127 r.fRight = rect.mRight;
128 r.fBottom = rect.mBottom;
129
130 SkIRect ir;
131 r.roundOut(&ir);
132 GraphicsJNI::irect_to_jrect(ir, env, bounds);
133}
134
Seigo Nonaka6b9b5162022-02-03 15:20:20 +0900135// Regular JNI
136static jlong nGetExtent(JNIEnv* env, jobject, jlong ptr, jcharArray javaText, jint start,
137 jint end) {
138 ScopedCharArrayRO text(env, javaText);
139 const minikin::U16StringPiece textBuffer(text.get(), text.size());
140 const minikin::Range range(start, end);
141
142 minikin::MinikinExtent extent = toMeasuredParagraph(ptr)->getExtent(textBuffer, range);
143
144 int32_t ascent = SkScalarRoundToInt(extent.ascent);
145 int32_t descent = SkScalarRoundToInt(extent.descent);
146
147 return (((jlong)(ascent)) << 32) | ((jlong)descent);
148}
149
Seigo Nonaka783f9612018-01-20 12:11:13 -0800150// CriticalNative
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +0100151static jlong nGetReleaseFunc(CRITICAL_JNI_PARAMS) {
Seigo Nonaka9d3bd082018-01-11 10:02:12 -0800152 return toJLong(&releaseMeasuredParagraph);
Seigo Nonaka6f1868b2017-12-01 16:24:19 -0800153}
154
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +0100155static jint nGetMemoryUsage(CRITICAL_JNI_PARAMS_COMMA jlong ptr) {
Seigo Nonaka49ca0242018-01-24 16:46:14 -0800156 return static_cast<jint>(toMeasuredParagraph(ptr)->getMemoryUsage());
157}
158
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700159static const JNINativeMethod gMTBuilderMethods[] = {
Seigo Nonaka0f761402021-08-30 01:27:10 +0000160 // MeasuredParagraphBuilder native functions.
161 {"nInitBuilder", "()J", (void*)nInitBuilder},
James.cf Lin664a75b2022-01-19 16:56:34 +0800162 {"nAddStyleRun", "(JJIIIIZ)V", (void*)nAddStyleRun},
Seigo Nonaka0f761402021-08-30 01:27:10 +0000163 {"nAddReplacementRun", "(JJIIF)V", (void*)nAddReplacementRun},
Seigo Nonaka6e39daf2023-08-19 11:36:26 +0900164 {"nBuildMeasuredText", "(JJ[CZZZZ)J", (void*)nBuildMeasuredText},
Seigo Nonaka0f761402021-08-30 01:27:10 +0000165 {"nFreeBuilder", "(J)V", (void*)nFreeBuilder},
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700166};
Seigo Nonaka6f1868b2017-12-01 16:24:19 -0800167
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700168static const JNINativeMethod gMTMethods[] = {
Seigo Nonaka6b9b5162022-02-03 15:20:20 +0900169 // MeasuredParagraph native functions.
170 {"nGetWidth", "(JII)F", (void*)nGetWidth}, // Critical Natives
171 {"nGetBounds", "(J[CIILandroid/graphics/Rect;)V", (void*)nGetBounds}, // Regular JNI
172 {"nGetExtent", "(J[CII)J", (void*)nGetExtent}, // Regular JNI
173 {"nGetReleaseFunc", "()J", (void*)nGetReleaseFunc}, // Critical Natives
174 {"nGetMemoryUsage", "(J)I", (void*)nGetMemoryUsage}, // Critical Native
175 {"nGetCharWidthAt", "(JI)F", (void*)nGetCharWidthAt}, // Critical Native
Seigo Nonaka6f1868b2017-12-01 16:24:19 -0800176};
177
Seigo Nonaka70200b02018-10-01 16:04:11 -0700178int register_android_graphics_text_MeasuredText(JNIEnv* env) {
179 return RegisterMethodsOrDie(env, "android/graphics/text/MeasuredText",
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700180 gMTMethods, NELEM(gMTMethods))
Seigo Nonaka70200b02018-10-01 16:04:11 -0700181 + RegisterMethodsOrDie(env, "android/graphics/text/MeasuredText$Builder",
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700182 gMTBuilderMethods, NELEM(gMTBuilderMethods));
Seigo Nonaka6f1868b2017-12-01 16:24:19 -0800183}
184
185}