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