1 /*
2 * Copyright 2021 Google Inc.
3 *
4 * Use of this source code is governed by a BSD-style license that can be
5 * found in the LICENSE file.
6 */
7
8 #include "include/core/SkCanvas.h"
9 #include "include/core/SkColor.h"
10 #include "include/core/SkPaint.h"
11 #include "include/core/SkRRect.h"
12 #include "include/core/SkShader.h"
13 #include "modules/androidkit/src/Utils.h"
14
15 #include <jni.h>
16
17 namespace {
18
Canvas_GetWidth(JNIEnv * env,jobject,jlong native_instance)19 jint Canvas_GetWidth(JNIEnv* env, jobject, jlong native_instance) {
20 const auto* canvas = reinterpret_cast<const SkCanvas*>(native_instance);
21 return canvas ? canvas->imageInfo().width() : 0;
22 }
23
Canvas_GetHeight(JNIEnv * env,jobject,jlong native_instance)24 jint Canvas_GetHeight(JNIEnv* env, jobject, jlong native_instance) {
25 const auto* canvas = reinterpret_cast<const SkCanvas*>(native_instance);
26 return canvas ? canvas->imageInfo().height() : 0;
27 }
28
Canvas_Save(JNIEnv * env,jobject,jlong native_instance)29 jint Canvas_Save(JNIEnv* env, jobject, jlong native_instance) {
30 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
31 return canvas->save();
32 }
33 return 0;
34 }
35
Canvas_Restore(JNIEnv * env,jobject,jlong native_instance)36 void Canvas_Restore(JNIEnv* env, jobject, jlong native_instance) {
37 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
38 canvas->restore();
39 }
40 }
41
Canvas_RestoreToCount(JNIEnv * env,jobject,jlong native_instance,jint count)42 void Canvas_RestoreToCount(JNIEnv* env, jobject, jlong native_instance, jint count) {
43 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
44 canvas->restoreToCount(count);
45 }
46 }
47
Canvas_SaveLayer(JNIEnv * env,jobject,jlong native_instance,jlong native_paint)48 jint Canvas_SaveLayer(JNIEnv* env, jobject, jlong native_instance, jlong native_paint) {
49 auto* paint = reinterpret_cast<SkPaint* >(native_paint);
50 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
51 return canvas->saveLayer(nullptr, paint);
52 }
53 return 0;
54 }
55
Canvas_LocalToDevice(JNIEnv * env,jobject,jlong native_instance)56 jlong Canvas_LocalToDevice(JNIEnv* env, jobject, jlong native_instance) {
57 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
58 SkM44* m = new SkM44(canvas->getLocalToDevice());
59 return reinterpret_cast<jlong>(m);
60 }
61 return 0;
62 }
63
Canvas_Concat(JNIEnv * env,jobject,jlong native_instance,jlong native_matrix)64 void Canvas_Concat(JNIEnv* env, jobject, jlong native_instance, jlong native_matrix) {
65 auto* canvas = reinterpret_cast<SkCanvas*>(native_instance);
66 auto* matrix = reinterpret_cast<SkM44* >(native_matrix);
67
68 if (canvas && matrix) {
69 canvas->concat(*matrix);
70 }
71 }
72
Canvas_Concat16f(JNIEnv * env,jobject,jlong native_instance,jfloatArray jmatrix)73 void Canvas_Concat16f(JNIEnv* env, jobject, jlong native_instance, jfloatArray jmatrix) {
74 SkASSERT(env->GetArrayLength(jmatrix) == 16);
75
76 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
77 auto* m = env->GetFloatArrayElements(jmatrix, nullptr);
78 canvas->concat(SkM44::RowMajor(m));
79 env->ReleaseFloatArrayElements(jmatrix, m, 0);
80 }
81 }
82
Canvas_Translate(JNIEnv * env,jobject,jlong native_instance,jfloat tx,jfloat ty,jfloat tz)83 void Canvas_Translate(JNIEnv* env, jobject, jlong native_instance,
84 jfloat tx, jfloat ty, jfloat tz) {
85 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
86 canvas->concat(SkM44::Translate(tx, ty, tz));
87 }
88 }
89
Canvas_Scale(JNIEnv * env,jobject,jlong native_instance,jfloat sx,jfloat sy,jfloat sz)90 void Canvas_Scale(JNIEnv* env, jobject, jlong native_instance, jfloat sx, jfloat sy, jfloat sz) {
91 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
92 canvas->concat(SkM44::Scale(sx, sy, sz));
93 }
94 }
95
Canvas_ClipPath(JNIEnv * env,jobject,jlong native_instance,jlong native_path,jint native_clipOp,jboolean doAA)96 void Canvas_ClipPath(JNIEnv* env, jobject, jlong native_instance, jlong native_path,
97 jint native_clipOp, jboolean doAA) {
98 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
99 if (auto* path = reinterpret_cast<SkPath*>(native_path)) {
100 canvas->clipPath(*path, static_cast<SkClipOp>(native_clipOp), doAA);
101 }
102 }
103 }
104
Canvas_ClipRect(JNIEnv * env,jobject,jlong native_instance,jfloat l,jfloat t,jfloat r,jfloat b,jint native_clipOp,jboolean doAA)105 void Canvas_ClipRect(JNIEnv* env, jobject, jlong native_instance, jfloat l, jfloat t, jfloat r, jfloat b,
106 jint native_clipOp, jboolean doAA) {
107 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
108 canvas->clipRect(SkRect::MakeLTRB(l, t, r, b), static_cast<SkClipOp>(native_clipOp), doAA);
109 }
110 }
111
Canvas_ClipRRect(JNIEnv * env,jobject,jlong native_instance,jfloat l,jfloat t,jfloat r,jfloat b,jfloat xRad,jfloat yRad,jint native_clipOp,jboolean doAA)112 void Canvas_ClipRRect(JNIEnv* env, jobject, jlong native_instance, jfloat l, jfloat t, jfloat r, jfloat b,
113 jfloat xRad, jfloat yRad,
114 jint native_clipOp, jboolean doAA) {
115 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
116 canvas->clipRRect(SkRRect::MakeRectXY(SkRect::MakeLTRB(l, t, r, b), xRad, yRad),
117 static_cast<SkClipOp>(native_clipOp), doAA);
118 }
119 }
120
Canvas_ClipShader(JNIEnv * env,jobject,jlong native_instance,jlong native_shader,jint native_clipOp)121 void Canvas_ClipShader(JNIEnv* env, jobject, jlong native_instance, jlong native_shader, jint native_clipOp) {
122 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
123 if (auto* shader = reinterpret_cast<SkShader*>(native_shader)) {
124 canvas->clipShader(sk_ref_sp(shader), static_cast<SkClipOp>(native_clipOp));
125 }
126 }
127 }
128
Canvas_DrawColor(JNIEnv * env,jobject,jlong native_instance,float r,float g,float b,float a)129 void Canvas_DrawColor(JNIEnv* env, jobject, jlong native_instance,
130 float r, float g, float b, float a) {
131 if (auto* canvas = reinterpret_cast<SkCanvas*>(native_instance)) {
132 canvas->drawColor(SkColor4f{r, g, b, a});
133 }
134 }
135
Canvas_DrawRect(JNIEnv * env,jobject,jlong native_instance,jfloat left,jfloat top,jfloat right,jfloat bottom,jlong native_paint)136 void Canvas_DrawRect(JNIEnv* env, jobject, jlong native_instance,
137 jfloat left, jfloat top, jfloat right, jfloat bottom,
138 jlong native_paint) {
139 auto* canvas = reinterpret_cast<SkCanvas*>(native_instance);
140 auto* paint = reinterpret_cast<SkPaint* >(native_paint);
141 if (canvas && paint) {
142 canvas->drawRect(SkRect::MakeLTRB(left, top, right, bottom), *paint);
143 }
144 }
145
Canvas_DrawImage(JNIEnv * env,jobject,jlong native_instance,jlong native_image,jfloat x,jfloat y,jint sampling_desc,jfloat sampling_b,jfloat sampling_c)146 void Canvas_DrawImage(JNIEnv* env, jobject, jlong native_instance, jlong native_image,
147 jfloat x, jfloat y,
148 jint sampling_desc, jfloat sampling_b, jfloat sampling_c) {
149 auto* canvas = reinterpret_cast<SkCanvas*>(native_instance);
150 auto* image = reinterpret_cast<SkImage *>(native_image);
151
152 if (canvas && image) {
153 canvas->drawImage(image, x, y,
154 androidkit::utils::SamplingOptions(sampling_desc, sampling_b, sampling_c));
155 }
156 }
157
Canvas_DrawPath(JNIEnv * env,jobject,jlong native_instance,jlong native_path,jlong native_paint)158 void Canvas_DrawPath(JNIEnv* env, jobject, jlong native_instance, jlong native_path,
159 jlong native_paint) {
160 auto* canvas = reinterpret_cast<SkCanvas*>(native_instance);
161 auto* path = reinterpret_cast<SkPath* >(native_path);
162 auto* paint = reinterpret_cast<SkPaint* >(native_paint);
163 if (canvas && paint && path) {
164 canvas->drawPath(*path, *paint);
165 }
166 }
167
168 // jPos: a composite array in the form of [x1, y1, x2, y2, ... ,xn, yn]
169 // callers of this function check should throw IllegalArgumentException in Java
Canvas_DrawGlyphs(JNIEnv * env,jobject,jlong native_instance,jcharArray jglyphs,jfloatArray jPos,jfloat xOrigin,jfloat yOrigin,jlong native_font,jlong native_paint)170 void Canvas_DrawGlyphs(JNIEnv* env, jobject, jlong native_instance, jcharArray jglyphs,
171 jfloatArray jPos, jfloat xOrigin, jfloat yOrigin,
172 jlong native_font, jlong native_paint) {
173 auto* canvas = reinterpret_cast<SkCanvas*>(native_instance);
174 auto* font = reinterpret_cast<SkFont*>(native_font);
175 auto* paint = reinterpret_cast<SkPaint* >(native_paint);
176 if (canvas && font && paint) {
177 int count = env->GetArrayLength(jglyphs);
178 auto* compositePositions = env->GetFloatArrayElements(jPos, nullptr);
179 auto* positions = reinterpret_cast<SkPoint*>(compositePositions);
180 auto* glyphs = env->GetCharArrayElements(jglyphs, nullptr);
181 canvas->drawGlyphs(count, glyphs, positions, {xOrigin, yOrigin}, *font, *paint);
182
183 env->ReleaseCharArrayElements(jglyphs, glyphs, 0);
184 env->ReleaseFloatArrayElements(jPos, compositePositions, 0);
185 }
186 }
187
188 } // namespace
189
register_androidkit_Canvas(JNIEnv * env)190 int register_androidkit_Canvas(JNIEnv* env) {
191 static const JNINativeMethod methods[] = {
192 {"nGetWidth" , "(J)I" , reinterpret_cast<void*>(Canvas_GetWidth) },
193 {"nGetHeight" , "(J)I" , reinterpret_cast<void*>(Canvas_GetHeight) },
194 {"nSave" , "(J)I" , reinterpret_cast<void*>(Canvas_Save) },
195 {"nSaveLayer" , "(JJ)I" , reinterpret_cast<void*>(Canvas_SaveLayer) },
196 {"nRestore" , "(J)V" , reinterpret_cast<void*>(Canvas_Restore) },
197 {"nRestoreToCount" , "(JI)V" , reinterpret_cast<void*>(Canvas_RestoreToCount)},
198 {"nGetLocalToDevice", "(J)J" , reinterpret_cast<void*>(Canvas_LocalToDevice) },
199 {"nConcat" , "(JJ)V" , reinterpret_cast<void*>(Canvas_Concat) },
200 {"nConcat16f" , "(J[F)V" , reinterpret_cast<void*>(Canvas_Concat16f) },
201 {"nTranslate" , "(JFFF)V" , reinterpret_cast<void*>(Canvas_Translate) },
202 {"nScale" , "(JFFF)V" , reinterpret_cast<void*>(Canvas_Scale) },
203 {"nClipPath" , "(JJIZ)V" , reinterpret_cast<void*>(Canvas_ClipPath) },
204 {"nClipRect" , "(JFFFFIZ)V" , reinterpret_cast<void*>(Canvas_ClipRect) },
205 {"nClipRRect" , "(JFFFFFFIZ)V" , reinterpret_cast<void*>(Canvas_ClipRRect) },
206 {"nClipShader" , "(JJI)V" , reinterpret_cast<void*>(Canvas_ClipShader) },
207 {"nDrawColor" , "(JFFFF)V" , reinterpret_cast<void*>(Canvas_DrawColor) },
208 {"nDrawRect" , "(JFFFFJ)V" , reinterpret_cast<void*>(Canvas_DrawRect) },
209 {"nDrawImage" , "(JJFFIFF)V" , reinterpret_cast<void*>(Canvas_DrawImage) },
210 {"nDrawPath" , "(JJJ)V" , reinterpret_cast<void*>(Canvas_DrawPath) },
211 {"nDrawGlyphs" , "(J[C[FFFJJ)V", reinterpret_cast<void*>(Canvas_DrawGlyphs) },
212
213 };
214
215 const auto clazz = env->FindClass("org/skia/androidkit/Canvas");
216 return clazz
217 ? env->RegisterNatives(clazz, methods, SK_ARRAY_COUNT(methods))
218 : JNI_ERR;
219 }
220