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