blob: 746745afbf09bb9a0b4fe932824bf85d3af8a186 [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,
Seigo Nonaka6c9956e2023-06-29 17:45:00 +090065 jlong paintPtr, jint lbStyle, jint lbWordStyle, jboolean hyphenation,
66 jint start, jint end, 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)
Seigo Nonaka6c9956e2023-06-29 17:45:00 +090071 ->addStyleRun(start, end, std::move(minikinPaint), lbStyle, lbWordStyle, hyphenation,
72 isRtl);
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080073}
74
75// Regular JNI
76static void nAddReplacementRun(JNIEnv* /* unused */, jclass /* unused */, jlong builderPtr,
77 jlong paintPtr, jint start, jint end, jfloat width) {
78 toBuilder(builderPtr)->addReplacementRun(start, end, width,
79 toPaint(paintPtr)->getMinikinLocaleListId());
80}
81
82// Regular JNI
Seigo Nonaka0f761402021-08-30 01:27:10 +000083static jlong nBuildMeasuredText(JNIEnv* env, jclass /* unused */, jlong builderPtr, jlong hintPtr,
84 jcharArray javaText, jboolean computeHyphenation,
Seigo Nonaka6e39daf2023-08-19 11:36:26 +090085 jboolean computeLayout, jboolean computeBounds,
86 jboolean fastHyphenationMode) {
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080087 ScopedCharArrayRO text(env, javaText);
88 const minikin::U16StringPiece textBuffer(text.get(), text.size());
89
90 // Pass the ownership to Java.
Seigo Nonaka0f761402021-08-30 01:27:10 +000091 return toJLong(toBuilder(builderPtr)
Seigo Nonaka6e39daf2023-08-19 11:36:26 +090092 ->build(textBuffer, computeHyphenation, computeLayout, computeBounds,
Seigo Nonaka0f761402021-08-30 01:27:10 +000093 fastHyphenationMode, toMeasuredParagraph(hintPtr))
94 .release());
Seigo Nonaka6f1868b2017-12-01 16:24:19 -080095}
96
97// Regular JNI
98static void nFreeBuilder(JNIEnv* env, jclass /* unused */, jlong builderPtr) {
99 delete toBuilder(builderPtr);
100}
101
102// CriticalNative
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +0100103static jfloat nGetWidth(CRITICAL_JNI_PARAMS_COMMA jlong ptr, jint start, jint end) {
Seigo Nonaka783f9612018-01-20 12:11:13 -0800104 minikin::MeasuredText* mt = toMeasuredParagraph(ptr);
105 float r = 0.0f;
106 for (int i = start; i < end; ++i) {
107 r += mt->widths[i];
108 }
109 return r;
110}
111
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +0100112static jfloat nGetCharWidthAt(CRITICAL_JNI_PARAMS_COMMA jlong ptr, jint offset) {
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700113 return toMeasuredParagraph(ptr)->widths[offset];
114}
115
Seigo Nonakaa5534772018-03-15 00:22:20 -0700116// Regular JNI
Seigo Nonakafb0abe12018-04-02 23:25:38 -0700117static void nGetBounds(JNIEnv* env, jobject, jlong ptr, jcharArray javaText, jint start, jint end,
118 jobject bounds) {
Seigo Nonakaa5534772018-03-15 00:22:20 -0700119 ScopedCharArrayRO text(env, javaText);
120 const minikin::U16StringPiece textBuffer(text.get(), text.size());
Seigo Nonakafb0abe12018-04-02 23:25:38 -0700121 const minikin::Range range(start, end);
Seigo Nonakaa5534772018-03-15 00:22:20 -0700122
Seigo Nonakafb0abe12018-04-02 23:25:38 -0700123 minikin::MinikinRect rect = toMeasuredParagraph(ptr)->getBounds(textBuffer, range);
Seigo Nonakaa5534772018-03-15 00:22:20 -0700124
125 SkRect r;
126 r.fLeft = rect.mLeft;
127 r.fTop = rect.mTop;
128 r.fRight = rect.mRight;
129 r.fBottom = rect.mBottom;
130
131 SkIRect ir;
132 r.roundOut(&ir);
133 GraphicsJNI::irect_to_jrect(ir, env, bounds);
134}
135
Seigo Nonaka6b9b5162022-02-03 15:20:20 +0900136// Regular JNI
137static jlong nGetExtent(JNIEnv* env, jobject, jlong ptr, jcharArray javaText, jint start,
138 jint end) {
139 ScopedCharArrayRO text(env, javaText);
140 const minikin::U16StringPiece textBuffer(text.get(), text.size());
141 const minikin::Range range(start, end);
142
143 minikin::MinikinExtent extent = toMeasuredParagraph(ptr)->getExtent(textBuffer, range);
144
145 int32_t ascent = SkScalarRoundToInt(extent.ascent);
146 int32_t descent = SkScalarRoundToInt(extent.descent);
147
148 return (((jlong)(ascent)) << 32) | ((jlong)descent);
149}
150
Seigo Nonaka783f9612018-01-20 12:11:13 -0800151// CriticalNative
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +0100152static jlong nGetReleaseFunc(CRITICAL_JNI_PARAMS) {
Seigo Nonaka9d3bd082018-01-11 10:02:12 -0800153 return toJLong(&releaseMeasuredParagraph);
Seigo Nonaka6f1868b2017-12-01 16:24:19 -0800154}
155
Jerome Gaillard21e7e2d2019-05-14 14:34:46 +0100156static jint nGetMemoryUsage(CRITICAL_JNI_PARAMS_COMMA jlong ptr) {
Seigo Nonaka49ca0242018-01-24 16:46:14 -0800157 return static_cast<jint>(toMeasuredParagraph(ptr)->getMemoryUsage());
158}
159
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700160static const JNINativeMethod gMTBuilderMethods[] = {
Seigo Nonaka0f761402021-08-30 01:27:10 +0000161 // MeasuredParagraphBuilder native functions.
162 {"nInitBuilder", "()J", (void*)nInitBuilder},
Seigo Nonaka6c9956e2023-06-29 17:45:00 +0900163 {"nAddStyleRun", "(JJIIZIIZ)V", (void*)nAddStyleRun},
Seigo Nonaka0f761402021-08-30 01:27:10 +0000164 {"nAddReplacementRun", "(JJIIF)V", (void*)nAddReplacementRun},
Seigo Nonaka6e39daf2023-08-19 11:36:26 +0900165 {"nBuildMeasuredText", "(JJ[CZZZZ)J", (void*)nBuildMeasuredText},
Seigo Nonaka0f761402021-08-30 01:27:10 +0000166 {"nFreeBuilder", "(J)V", (void*)nFreeBuilder},
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700167};
Seigo Nonaka6f1868b2017-12-01 16:24:19 -0800168
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700169static const JNINativeMethod gMTMethods[] = {
Seigo Nonaka6b9b5162022-02-03 15:20:20 +0900170 // MeasuredParagraph native functions.
171 {"nGetWidth", "(JII)F", (void*)nGetWidth}, // Critical Natives
172 {"nGetBounds", "(J[CIILandroid/graphics/Rect;)V", (void*)nGetBounds}, // Regular JNI
173 {"nGetExtent", "(J[CII)J", (void*)nGetExtent}, // Regular JNI
174 {"nGetReleaseFunc", "()J", (void*)nGetReleaseFunc}, // Critical Natives
175 {"nGetMemoryUsage", "(J)I", (void*)nGetMemoryUsage}, // Critical Native
176 {"nGetCharWidthAt", "(JI)F", (void*)nGetCharWidthAt}, // Critical Native
Seigo Nonaka6f1868b2017-12-01 16:24:19 -0800177};
178
Seigo Nonaka70200b02018-10-01 16:04:11 -0700179int register_android_graphics_text_MeasuredText(JNIEnv* env) {
180 return RegisterMethodsOrDie(env, "android/graphics/text/MeasuredText",
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700181 gMTMethods, NELEM(gMTMethods))
Seigo Nonaka70200b02018-10-01 16:04:11 -0700182 + RegisterMethodsOrDie(env, "android/graphics/text/MeasuredText$Builder",
Seigo Nonaka6f11c6e2018-07-24 11:26:18 -0700183 gMTBuilderMethods, NELEM(gMTBuilderMethods));
Seigo Nonaka6f1868b2017-12-01 16:24:19 -0800184}
185
186}