Matrix.cpp revision 2e0103eb340822f9d580c1aa8492bae8394b8243
1/* libs/android_runtime/android/graphics/Matrix.cpp 2** 3** Copyright 2006, The Android Open Source Project 4** 5** Licensed under the Apache License, Version 2.0 (the "License"); 6** you may not use this file except in compliance with the License. 7** You may obtain a copy of the License at 8** 9** http://www.apache.org/licenses/LICENSE-2.0 10** 11** Unless required by applicable law or agreed to in writing, software 12** distributed under the License is distributed on an "AS IS" BASIS, 13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 14** See the License for the specific language governing permissions and 15** limitations under the License. 16*/ 17 18#include "jni.h" 19#include "GraphicsJNI.h" 20#include <android_runtime/AndroidRuntime.h> 21 22#include "SkMatrix.h" 23#include "SkTemplates.h" 24 25#include "Matrix.h" 26 27#include <Caches.h> 28 29namespace android { 30 31class SkMatrixGlue { 32public: 33 34 static void finalizer(JNIEnv* env, jobject clazz, jlong objHandle) { 35 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 36 delete obj; 37 } 38 39 static jlong create(JNIEnv* env, jobject clazz, jlong srcHandle) { 40 const SkMatrix* src = reinterpret_cast<SkMatrix*>(srcHandle); 41 SkMatrix* obj = new SkMatrix(); 42 if (src) 43 *obj = *src; 44 else 45 obj->reset(); 46 return reinterpret_cast<jlong>(obj); 47 } 48 49 static jboolean isIdentity(JNIEnv* env, jobject clazz, jlong objHandle) { 50 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 51 return obj->isIdentity() ? JNI_TRUE : JNI_FALSE; 52 } 53 static jboolean rectStaysRect(JNIEnv* env, jobject clazz, jlong objHandle) { 54 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 55 return obj->rectStaysRect() ? JNI_TRUE : JNI_FALSE; 56 } 57 static void reset(JNIEnv* env, jobject clazz, jlong objHandle) { 58 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 59 obj->reset(); 60 } 61 static void set(JNIEnv* env, jobject clazz, jlong objHandle, jlong otherHandle) { 62 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 63 SkMatrix* other = reinterpret_cast<SkMatrix*>(otherHandle); 64 *obj = *other; 65 } 66 static void setTranslate(JNIEnv* env, jobject clazz, jlong objHandle, jfloat dx, jfloat dy) { 67 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 68 obj->setTranslate(dx, dy); 69 } 70 static void setScale__FFFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat sx, jfloat sy, jfloat px, jfloat py) { 71 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 72 obj->setScale(sx, sy, px, py); 73 } 74 static void setScale__FF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat sx, jfloat sy) { 75 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 76 obj->setScale(sx, sy); 77 } 78 static void setRotate__FFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat degrees, jfloat px, jfloat py) { 79 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 80 obj->setRotate(degrees, px, py); 81 } 82 static void setRotate__F(JNIEnv* env, jobject clazz, jlong objHandle, jfloat degrees) { 83 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 84 obj->setRotate(degrees); 85 } 86 static void setSinCos__FFFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat sinValue, jfloat cosValue, jfloat px, jfloat py) { 87 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 88 obj->setSinCos(sinValue, cosValue, px, py); 89 } 90 static void setSinCos__FF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat sinValue, jfloat cosValue) { 91 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 92 obj->setSinCos(sinValue, cosValue); 93 } 94 static void setSkew__FFFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat kx, jfloat ky, jfloat px, jfloat py) { 95 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 96 obj->setSkew(kx, ky, px, py); 97 } 98 static void setSkew__FF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat kx, jfloat ky) { 99 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 100 obj->setSkew(kx, ky); 101 } 102 static void setConcat(JNIEnv* env, jobject clazz, jlong objHandle, jlong aHandle, jlong bHandle) { 103 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 104 SkMatrix* a = reinterpret_cast<SkMatrix*>(aHandle); 105 SkMatrix* b = reinterpret_cast<SkMatrix*>(bHandle); 106 obj->setConcat(*a, *b); 107 } 108 109 static void preTranslate(JNIEnv* env, jobject clazz, jlong objHandle, jfloat dx, jfloat dy) { 110 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 111 obj->preTranslate(dx, dy); 112 } 113 114 static void preScale__FFFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat sx, jfloat sy, jfloat px, jfloat py) { 115 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 116 obj->preScale(sx, sy, px, py); 117 } 118 119 static void preScale__FF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat sx, jfloat sy) { 120 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 121 obj->preScale(sx, sy); 122 } 123 124 static void preRotate__FFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat degrees, jfloat px, jfloat py) { 125 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 126 obj->preRotate(degrees, px, py); 127 } 128 129 static void preRotate__F(JNIEnv* env, jobject clazz, jlong objHandle, jfloat degrees) { 130 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 131 obj->preRotate(degrees); 132 } 133 134 static void preSkew__FFFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat kx, jfloat ky, jfloat px, jfloat py) { 135 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 136 obj->preSkew(kx, ky, px, py); 137 } 138 139 static void preSkew__FF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat kx, jfloat ky) { 140 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 141 obj->preSkew(kx, ky); 142 } 143 144 static void preConcat(JNIEnv* env, jobject clazz, jlong objHandle, jlong otherHandle) { 145 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 146 SkMatrix* other = reinterpret_cast<SkMatrix*>(otherHandle); 147 obj->preConcat(*other); 148 } 149 150 static void postTranslate(JNIEnv* env, jobject clazz, jlong objHandle, jfloat dx, jfloat dy) { 151 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 152 obj->postTranslate(dx, dy); 153 } 154 155 static void postScale__FFFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat sx, jfloat sy, jfloat px, jfloat py) { 156 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 157 obj->postScale(sx, sy, px, py); 158 } 159 160 static void postScale__FF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat sx, jfloat sy) { 161 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 162 obj->postScale(sx, sy); 163 } 164 165 static void postRotate__FFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat degrees, jfloat px, jfloat py) { 166 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 167 obj->postRotate(degrees, px, py); 168 } 169 170 static void postRotate__F(JNIEnv* env, jobject clazz, jlong objHandle, jfloat degrees) { 171 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 172 obj->postRotate(degrees); 173 } 174 175 static void postSkew__FFFF(JNIEnv* env, jobject clazz, jlong objHandle, jfloat kx, jfloat ky, jfloat px, jfloat py) { 176 SkMatrix* obj = reinterpret_cast<SkMatrix*>(objHandle); 177 obj->postSkew(kx, ky, px, py); 178 } 179 180 static void postSkew__FF(JNIEnv* env, jobject clazz, jlong matrixHandle, jfloat kx, jfloat ky) { 181 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 182 matrix->postSkew(kx, ky); 183 } 184 185 static void postConcat(JNIEnv* env, jobject clazz, jlong matrixHandle, jlong otherHandle) { 186 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 187 SkMatrix* other = reinterpret_cast<SkMatrix*>(otherHandle); 188 matrix->postConcat(*other); 189 } 190 191 static jboolean setRectToRect(JNIEnv* env, jobject clazz, jlong matrixHandle, jobject src, jobject dst, jint stfHandle) { 192 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 193 SkMatrix::ScaleToFit stf = static_cast<SkMatrix::ScaleToFit>(stfHandle); 194 SkRect src_; 195 GraphicsJNI::jrectf_to_rect(env, src, &src_); 196 SkRect dst_; 197 GraphicsJNI::jrectf_to_rect(env, dst, &dst_); 198 return matrix->setRectToRect(src_, dst_, stf) ? JNI_TRUE : JNI_FALSE; 199 } 200 201 static jboolean setPolyToPoly(JNIEnv* env, jobject clazz, jlong matrixHandle, 202 jfloatArray jsrc, jint srcIndex, 203 jfloatArray jdst, jint dstIndex, jint ptCount) { 204 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 205 SkASSERT(srcIndex >= 0); 206 SkASSERT(dstIndex >= 0); 207 SkASSERT((unsigned)ptCount <= 4); 208 209 AutoJavaFloatArray autoSrc(env, jsrc, srcIndex + (ptCount << 1), kRO_JNIAccess); 210 AutoJavaFloatArray autoDst(env, jdst, dstIndex + (ptCount << 1), kRW_JNIAccess); 211 float* src = autoSrc.ptr() + srcIndex; 212 float* dst = autoDst.ptr() + dstIndex; 213 bool result; 214 215#ifdef SK_SCALAR_IS_FLOAT 216 result = matrix->setPolyToPoly((const SkPoint*)src, (const SkPoint*)dst, 217 ptCount); 218#else 219 SkASSERT(false); 220#endif 221 return result ? JNI_TRUE : JNI_FALSE; 222 } 223 224 static jboolean invert(JNIEnv* env, jobject clazz, jlong matrixHandle, jlong inverseHandle) { 225 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 226 SkMatrix* inverse = reinterpret_cast<SkMatrix*>(inverseHandle); 227 return matrix->invert(inverse); 228 } 229 230 static void mapPoints(JNIEnv* env, jobject clazz, jlong matrixHandle, 231 jfloatArray dst, jint dstIndex, 232 jfloatArray src, jint srcIndex, 233 jint ptCount, jboolean isPts) { 234 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 235 SkASSERT(ptCount >= 0); 236 AutoJavaFloatArray autoSrc(env, src, srcIndex + (ptCount << 1), kRO_JNIAccess); 237 AutoJavaFloatArray autoDst(env, dst, dstIndex + (ptCount << 1), kRW_JNIAccess); 238 float* srcArray = autoSrc.ptr() + srcIndex; 239 float* dstArray = autoDst.ptr() + dstIndex; 240#ifdef SK_SCALAR_IS_FLOAT 241 if (isPts) 242 matrix->mapPoints((SkPoint*)dstArray, (const SkPoint*)srcArray, 243 ptCount); 244 else 245 matrix->mapVectors((SkVector*)dstArray, (const SkVector*)srcArray, 246 ptCount); 247#else 248 SkASSERT(false); 249#endif 250 } 251 252 static jboolean mapRect__RectFRectF(JNIEnv* env, jobject clazz, jlong matrixHandle, jobjectArray dst, jobject src) { 253 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 254 SkRect dst_, src_; 255 GraphicsJNI::jrectf_to_rect(env, src, &src_); 256 jboolean rectStaysRect = matrix->mapRect(&dst_, src_); 257 GraphicsJNI::rect_to_jrectf(dst_, env, dst); 258 return rectStaysRect ? JNI_TRUE : JNI_FALSE; 259 } 260 261 static jfloat mapRadius(JNIEnv* env, jobject clazz, jlong matrixHandle, jfloat radius) { 262 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 263 float result; 264 result = SkScalarToFloat(matrix->mapRadius(radius)); 265 return static_cast<jfloat>(result); 266 } 267 static void getValues(JNIEnv* env, jobject clazz, jlong matrixHandle, jfloatArray values) { 268 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 269 AutoJavaFloatArray autoValues(env, values, 9, kRW_JNIAccess); 270 float* dst = autoValues.ptr(); 271#ifdef SK_SCALAR_IS_FLOAT 272 for (int i = 0; i < 9; i++) { 273 dst[i] = matrix->get(i); 274 } 275#else 276 SkASSERT(false); 277#endif 278 } 279 280 static void setValues(JNIEnv* env, jobject clazz, jlong matrixHandle, jfloatArray values) { 281 SkMatrix* matrix = reinterpret_cast<SkMatrix*>(matrixHandle); 282 AutoJavaFloatArray autoValues(env, values, 9, kRO_JNIAccess); 283 const float* src = autoValues.ptr(); 284 285#ifdef SK_SCALAR_IS_FLOAT 286 for (int i = 0; i < 9; i++) { 287 matrix->set(i, src[i]); 288 } 289#else 290 SkASSERT(false); 291#endif 292 } 293 294 static jboolean equals(JNIEnv* env, jobject clazz, jlong aHandle, jlong bHandle) { 295 const SkMatrix* a = reinterpret_cast<SkMatrix*>(aHandle); 296 const SkMatrix* b = reinterpret_cast<SkMatrix*>(bHandle); 297 return *a == *b; 298 } 299 }; 300 301static JNINativeMethod methods[] = { 302 {"finalizer", "(J)V", (void*) SkMatrixGlue::finalizer}, 303 {"native_create","(J)J", (void*) SkMatrixGlue::create}, 304 {"native_isIdentity","(J)Z", (void*) SkMatrixGlue::isIdentity}, 305 {"native_rectStaysRect","(J)Z", (void*) SkMatrixGlue::rectStaysRect}, 306 {"native_reset","(J)V", (void*) SkMatrixGlue::reset}, 307 {"native_set","(JJ)V", (void*) SkMatrixGlue::set}, 308 {"native_setTranslate","(JFF)V", (void*) SkMatrixGlue::setTranslate}, 309 {"native_setScale","(JFFFF)V", (void*) SkMatrixGlue::setScale__FFFF}, 310 {"native_setScale","(JFF)V", (void*) SkMatrixGlue::setScale__FF}, 311 {"native_setRotate","(JFFF)V", (void*) SkMatrixGlue::setRotate__FFF}, 312 {"native_setRotate","(JF)V", (void*) SkMatrixGlue::setRotate__F}, 313 {"native_setSinCos","(JFFFF)V", (void*) SkMatrixGlue::setSinCos__FFFF}, 314 {"native_setSinCos","(JFF)V", (void*) SkMatrixGlue::setSinCos__FF}, 315 {"native_setSkew","(JFFFF)V", (void*) SkMatrixGlue::setSkew__FFFF}, 316 {"native_setSkew","(JFF)V", (void*) SkMatrixGlue::setSkew__FF}, 317 {"native_setConcat","(JJJ)V", (void*) SkMatrixGlue::setConcat}, 318 {"native_preTranslate","(JFF)V", (void*) SkMatrixGlue::preTranslate}, 319 {"native_preScale","(JFFFF)V", (void*) SkMatrixGlue::preScale__FFFF}, 320 {"native_preScale","(JFF)V", (void*) SkMatrixGlue::preScale__FF}, 321 {"native_preRotate","(JFFF)V", (void*) SkMatrixGlue::preRotate__FFF}, 322 {"native_preRotate","(JF)V", (void*) SkMatrixGlue::preRotate__F}, 323 {"native_preSkew","(JFFFF)V", (void*) SkMatrixGlue::preSkew__FFFF}, 324 {"native_preSkew","(JFF)V", (void*) SkMatrixGlue::preSkew__FF}, 325 {"native_preConcat","(JJ)V", (void*) SkMatrixGlue::preConcat}, 326 {"native_postTranslate","(JFF)V", (void*) SkMatrixGlue::postTranslate}, 327 {"native_postScale","(JFFFF)V", (void*) SkMatrixGlue::postScale__FFFF}, 328 {"native_postScale","(JFF)V", (void*) SkMatrixGlue::postScale__FF}, 329 {"native_postRotate","(JFFF)V", (void*) SkMatrixGlue::postRotate__FFF}, 330 {"native_postRotate","(JF)V", (void*) SkMatrixGlue::postRotate__F}, 331 {"native_postSkew","(JFFFF)V", (void*) SkMatrixGlue::postSkew__FFFF}, 332 {"native_postSkew","(JFF)V", (void*) SkMatrixGlue::postSkew__FF}, 333 {"native_postConcat","(JJ)V", (void*) SkMatrixGlue::postConcat}, 334 {"native_setRectToRect","(JLandroid/graphics/RectF;Landroid/graphics/RectF;I)Z", (void*) SkMatrixGlue::setRectToRect}, 335 {"native_setPolyToPoly","(J[FI[FII)Z", (void*) SkMatrixGlue::setPolyToPoly}, 336 {"native_invert","(JJ)Z", (void*) SkMatrixGlue::invert}, 337 {"native_mapPoints","(J[FI[FIIZ)V", (void*) SkMatrixGlue::mapPoints}, 338 {"native_mapRect","(JLandroid/graphics/RectF;Landroid/graphics/RectF;)Z", (void*) SkMatrixGlue::mapRect__RectFRectF}, 339 {"native_mapRadius","(JF)F", (void*) SkMatrixGlue::mapRadius}, 340 {"native_getValues","(J[F)V", (void*) SkMatrixGlue::getValues}, 341 {"native_setValues","(J[F)V", (void*) SkMatrixGlue::setValues}, 342 {"native_equals", "(JJ)Z", (void*) SkMatrixGlue::equals} 343}; 344 345static jfieldID sNativeInstanceField; 346 347int register_android_graphics_Matrix(JNIEnv* env) { 348 int result = AndroidRuntime::registerNativeMethods(env, "android/graphics/Matrix", methods, 349 sizeof(methods) / sizeof(methods[0])); 350 351 jclass clazz = env->FindClass("android/graphics/Matrix"); 352 sNativeInstanceField = env->GetFieldID(clazz, "native_instance", "J"); 353 354 return result; 355} 356 357SkMatrix* android_graphics_Matrix_getSkMatrix(JNIEnv* env, jobject matrixObj) { 358 return reinterpret_cast<SkMatrix*>(env->GetLongField(matrixObj, sNativeInstanceField)); 359} 360 361} 362