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