1 /*
2 * Copyright (C) 2013 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
17 #include <Animator.h>
18 #include <Interpolator.h>
19 #include <RenderProperties.h>
20
21 #include "graphics_jni_helpers.h"
22
23 namespace android {
24
25 using namespace uirenderer;
26
27 static struct {
28 jclass clazz;
29
30 jmethodID callOnFinished;
31 } gRenderNodeAnimatorClassInfo;
32
getEnv(JavaVM * vm)33 static JNIEnv* getEnv(JavaVM* vm) {
34 JNIEnv* env;
35 if (vm->GetEnv(reinterpret_cast<void**>(&env), JNI_VERSION_1_6) != JNI_OK) {
36 return 0;
37 }
38 return env;
39 }
40
41 class AnimationListenerLifecycleChecker : public AnimationListener {
42 public:
onAnimationFinished(BaseRenderNodeAnimator * animator)43 virtual void onAnimationFinished(BaseRenderNodeAnimator* animator) {
44 LOG_ALWAYS_FATAL("Lifecycle failure, nStart(%p) wasn't called", animator);
45 }
46 };
47
48 static AnimationListenerLifecycleChecker sLifecycleChecker;
49
50 class AnimationListenerBridge : public AnimationListener {
51 public:
52 // This holds a strong reference to a Java WeakReference<T> object. This avoids
53 // cyclic-references-of-doom. If you think "I know, just use NewWeakGlobalRef!"
54 // then you end up with basically a PhantomReference, which is totally not
55 // what we want.
AnimationListenerBridge(JNIEnv * env,jobject finishListener)56 AnimationListenerBridge(JNIEnv* env, jobject finishListener) {
57 mFinishListener = env->NewGlobalRef(finishListener);
58 env->GetJavaVM(&mJvm);
59 }
60
~AnimationListenerBridge()61 virtual ~AnimationListenerBridge() {
62 if (mFinishListener) {
63 onAnimationFinished(NULL);
64 }
65 }
66
onAnimationFinished(BaseRenderNodeAnimator *)67 virtual void onAnimationFinished(BaseRenderNodeAnimator*) {
68 LOG_ALWAYS_FATAL_IF(!mFinishListener, "Finished listener twice?");
69 JNIEnv* env = getEnv(mJvm);
70 env->CallStaticVoidMethod(
71 gRenderNodeAnimatorClassInfo.clazz,
72 gRenderNodeAnimatorClassInfo.callOnFinished,
73 mFinishListener);
74 releaseJavaObject();
75 }
76
77 private:
releaseJavaObject()78 void releaseJavaObject() {
79 JNIEnv* env = getEnv(mJvm);
80 env->DeleteGlobalRef(mFinishListener);
81 mFinishListener = NULL;
82 }
83
84 JavaVM* mJvm;
85 jobject mFinishListener;
86 };
87
toRenderProperty(jint property)88 static inline RenderPropertyAnimator::RenderProperty toRenderProperty(jint property) {
89 LOG_ALWAYS_FATAL_IF(property < 0 || property > RenderPropertyAnimator::ALPHA,
90 "Invalid property %d", property);
91 return static_cast<RenderPropertyAnimator::RenderProperty>(property);
92 }
93
toPaintField(jint field)94 static inline CanvasPropertyPaintAnimator::PaintField toPaintField(jint field) {
95 LOG_ALWAYS_FATAL_IF(field < 0
96 || field > CanvasPropertyPaintAnimator::ALPHA,
97 "Invalid paint field %d", field);
98 return static_cast<CanvasPropertyPaintAnimator::PaintField>(field);
99 }
100
createAnimator(JNIEnv * env,jobject clazz,jint propertyRaw,jfloat finalValue)101 static jlong createAnimator(JNIEnv* env, jobject clazz,
102 jint propertyRaw, jfloat finalValue) {
103 RenderPropertyAnimator::RenderProperty property = toRenderProperty(propertyRaw);
104 BaseRenderNodeAnimator* animator = new RenderPropertyAnimator(property, finalValue);
105 animator->setListener(&sLifecycleChecker);
106 return reinterpret_cast<jlong>( animator );
107 }
108
createCanvasPropertyFloatAnimator(JNIEnv * env,jobject clazz,jlong canvasPropertyPtr,jfloat finalValue)109 static jlong createCanvasPropertyFloatAnimator(JNIEnv* env, jobject clazz,
110 jlong canvasPropertyPtr, jfloat finalValue) {
111 CanvasPropertyPrimitive* canvasProperty = reinterpret_cast<CanvasPropertyPrimitive*>(canvasPropertyPtr);
112 BaseRenderNodeAnimator* animator = new CanvasPropertyPrimitiveAnimator(canvasProperty, finalValue);
113 animator->setListener(&sLifecycleChecker);
114 return reinterpret_cast<jlong>( animator );
115 }
116
createCanvasPropertyPaintAnimator(JNIEnv * env,jobject clazz,jlong canvasPropertyPtr,jint paintFieldRaw,jfloat finalValue)117 static jlong createCanvasPropertyPaintAnimator(JNIEnv* env, jobject clazz,
118 jlong canvasPropertyPtr, jint paintFieldRaw,
119 jfloat finalValue) {
120 CanvasPropertyPaint* canvasProperty = reinterpret_cast<CanvasPropertyPaint*>(canvasPropertyPtr);
121 CanvasPropertyPaintAnimator::PaintField paintField = toPaintField(paintFieldRaw);
122 BaseRenderNodeAnimator* animator = new CanvasPropertyPaintAnimator(
123 canvasProperty, paintField, finalValue);
124 animator->setListener(&sLifecycleChecker);
125 return reinterpret_cast<jlong>( animator );
126 }
127
createRevealAnimator(JNIEnv * env,jobject clazz,jint centerX,jint centerY,jfloat startRadius,jfloat endRadius)128 static jlong createRevealAnimator(JNIEnv* env, jobject clazz,
129 jint centerX, jint centerY, jfloat startRadius, jfloat endRadius) {
130 BaseRenderNodeAnimator* animator = new RevealAnimator(centerX, centerY, startRadius, endRadius);
131 animator->setListener(&sLifecycleChecker);
132 return reinterpret_cast<jlong>( animator );
133 }
134
setStartValue(JNIEnv * env,jobject clazz,jlong animatorPtr,jfloat startValue)135 static void setStartValue(JNIEnv* env, jobject clazz, jlong animatorPtr, jfloat startValue) {
136 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
137 animator->setStartValue(startValue);
138 }
139
setDuration(JNIEnv * env,jobject clazz,jlong animatorPtr,jlong duration)140 static void setDuration(JNIEnv* env, jobject clazz, jlong animatorPtr, jlong duration) {
141 LOG_ALWAYS_FATAL_IF(duration < 0, "Duration cannot be negative");
142 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
143 animator->setDuration(duration);
144 }
145
getDuration(JNIEnv * env,jobject clazz,jlong animatorPtr)146 static jlong getDuration(JNIEnv* env, jobject clazz, jlong animatorPtr) {
147 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
148 return static_cast<jlong>(animator->duration());
149 }
150
setStartDelay(JNIEnv * env,jobject clazz,jlong animatorPtr,jlong startDelay)151 static void setStartDelay(JNIEnv* env, jobject clazz, jlong animatorPtr, jlong startDelay) {
152 LOG_ALWAYS_FATAL_IF(startDelay < 0, "Start delay cannot be negative");
153 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
154 animator->setStartDelay(startDelay);
155 }
156
setInterpolator(JNIEnv * env,jobject clazz,jlong animatorPtr,jlong interpolatorPtr)157 static void setInterpolator(JNIEnv* env, jobject clazz, jlong animatorPtr, jlong interpolatorPtr) {
158 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
159 Interpolator* interpolator = reinterpret_cast<Interpolator*>(interpolatorPtr);
160 animator->setInterpolator(interpolator);
161 }
162
setAllowRunningAsync(JNIEnv * env,jobject clazz,jlong animatorPtr,jboolean mayRunAsync)163 static void setAllowRunningAsync(JNIEnv* env, jobject clazz, jlong animatorPtr, jboolean mayRunAsync) {
164 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
165 animator->setAllowRunningAsync(mayRunAsync);
166 }
167
setListener(JNIEnv * env,jobject clazz,jlong animatorPtr,jobject finishListener)168 static void setListener(JNIEnv* env, jobject clazz, jlong animatorPtr, jobject finishListener) {
169 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
170 animator->setListener(new AnimationListenerBridge(env, finishListener));
171 }
172
start(JNIEnv * env,jobject clazz,jlong animatorPtr)173 static void start(JNIEnv* env, jobject clazz, jlong animatorPtr) {
174 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
175 animator->start();
176 }
177
end(JNIEnv * env,jobject clazz,jlong animatorPtr)178 static void end(JNIEnv* env, jobject clazz, jlong animatorPtr) {
179 BaseRenderNodeAnimator* animator = reinterpret_cast<BaseRenderNodeAnimator*>(animatorPtr);
180 animator->cancel();
181 }
182
183 // ----------------------------------------------------------------------------
184 // JNI Glue
185 // ----------------------------------------------------------------------------
186
187 const char* const kClassPathName = "android/graphics/animation/RenderNodeAnimator";
188
189 static const JNINativeMethod gMethods[] = {
190 { "nCreateAnimator", "(IF)J", (void*) createAnimator },
191 { "nCreateCanvasPropertyFloatAnimator", "(JF)J", (void*) createCanvasPropertyFloatAnimator },
192 { "nCreateCanvasPropertyPaintAnimator", "(JIF)J", (void*) createCanvasPropertyPaintAnimator },
193 { "nCreateRevealAnimator", "(IIFF)J", (void*) createRevealAnimator },
194 { "nSetStartValue", "(JF)V", (void*) setStartValue },
195 { "nSetDuration", "(JJ)V", (void*) setDuration },
196 { "nGetDuration", "(J)J", (void*) getDuration },
197 { "nSetStartDelay", "(JJ)V", (void*) setStartDelay },
198 { "nSetInterpolator", "(JJ)V", (void*) setInterpolator },
199 { "nSetAllowRunningAsync", "(JZ)V", (void*) setAllowRunningAsync },
200 { "nSetListener", "(JLandroid/graphics/animation/RenderNodeAnimator;)V", (void*) setListener},
201 { "nStart", "(J)V", (void*) start},
202 { "nEnd", "(J)V", (void*) end },
203 };
204
register_android_graphics_animation_RenderNodeAnimator(JNIEnv * env)205 int register_android_graphics_animation_RenderNodeAnimator(JNIEnv* env) {
206 sLifecycleChecker.incStrong(0);
207 gRenderNodeAnimatorClassInfo.clazz = FindClassOrDie(env, kClassPathName);
208 gRenderNodeAnimatorClassInfo.clazz = MakeGlobalRefOrDie(env,
209 gRenderNodeAnimatorClassInfo.clazz);
210
211 gRenderNodeAnimatorClassInfo.callOnFinished = GetStaticMethodIDOrDie(
212 env, gRenderNodeAnimatorClassInfo.clazz, "callOnFinished",
213 "(Landroid/graphics/animation/RenderNodeAnimator;)V");
214
215 return RegisterMethodsOrDie(env, kClassPathName, gMethods, NELEM(gMethods));
216 }
217
218
219 } // namespace android
220