1 /* 2 * Copyright (C) 2007 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 package android.view.animation; 18 19 import android.annotation.AnimRes; 20 import android.annotation.InterpolatorRes; 21 import android.annotation.TestApi; 22 import android.compat.annotation.UnsupportedAppUsage; 23 import android.content.Context; 24 import android.content.res.Resources; 25 import android.content.res.Resources.NotFoundException; 26 import android.content.res.Resources.Theme; 27 import android.content.res.XmlResourceParser; 28 import android.os.SystemClock; 29 import android.util.AttributeSet; 30 import android.util.Xml; 31 32 import org.xmlpull.v1.XmlPullParser; 33 import org.xmlpull.v1.XmlPullParserException; 34 35 import java.io.IOException; 36 37 /** 38 * Defines common utilities for working with animations. 39 * 40 */ 41 public class AnimationUtils { 42 43 /** 44 * These flags are used when parsing AnimatorSet objects 45 */ 46 private static final int TOGETHER = 0; 47 private static final int SEQUENTIALLY = 1; 48 49 private static class AnimationState { 50 boolean animationClockLocked; 51 long currentVsyncTimeMillis; 52 long lastReportedTimeMillis; 53 }; 54 55 private static ThreadLocal<AnimationState> sAnimationState 56 = new ThreadLocal<AnimationState>() { 57 @Override 58 protected AnimationState initialValue() { 59 return new AnimationState(); 60 } 61 }; 62 63 /** 64 * Locks AnimationUtils{@link #currentAnimationTimeMillis()} to a fixed value for the current 65 * thread. This is used by {@link android.view.Choreographer} to ensure that all accesses 66 * during a vsync update are synchronized to the timestamp of the vsync. 67 * 68 * It is also exposed to tests to allow for rapid, flake-free headless testing. 69 * 70 * Must be followed by a call to {@link #unlockAnimationClock()} to allow time to 71 * progress. Failing to do this will result in stuck animations, scrolls, and flings. 72 * 73 * Note that time is not allowed to "rewind" and must perpetually flow forward. So the 74 * lock may fail if the time is in the past from a previously returned value, however 75 * time will be frozen for the duration of the lock. The clock is a thread-local, so 76 * ensure that {@link #lockAnimationClock(long)}, {@link #unlockAnimationClock()}, and 77 * {@link #currentAnimationTimeMillis()} are all called on the same thread. 78 * 79 * This is also not reference counted in any way. Any call to {@link #unlockAnimationClock()} 80 * will unlock the clock for everyone on the same thread. It is therefore recommended 81 * for tests to use their own thread to ensure that there is no collision with any existing 82 * {@link android.view.Choreographer} instance. 83 * 84 * @hide 85 * */ 86 @TestApi lockAnimationClock(long vsyncMillis)87 public static void lockAnimationClock(long vsyncMillis) { 88 AnimationState state = sAnimationState.get(); 89 state.animationClockLocked = true; 90 state.currentVsyncTimeMillis = vsyncMillis; 91 } 92 93 /** 94 * Frees the time lock set in place by {@link #lockAnimationClock(long)}. Must be called 95 * to allow the animation clock to self-update. 96 * 97 * @hide 98 */ 99 @TestApi unlockAnimationClock()100 public static void unlockAnimationClock() { 101 sAnimationState.get().animationClockLocked = false; 102 } 103 104 /** 105 * Returns the current animation time in milliseconds. This time should be used when invoking 106 * {@link Animation#setStartTime(long)}. Refer to {@link android.os.SystemClock} for more 107 * information about the different available clocks. The clock used by this method is 108 * <em>not</em> the "wall" clock (it is not {@link System#currentTimeMillis}). 109 * 110 * @return the current animation time in milliseconds 111 * 112 * @see android.os.SystemClock 113 */ currentAnimationTimeMillis()114 public static long currentAnimationTimeMillis() { 115 AnimationState state = sAnimationState.get(); 116 if (state.animationClockLocked) { 117 // It's important that time never rewinds 118 return Math.max(state.currentVsyncTimeMillis, 119 state.lastReportedTimeMillis); 120 } 121 state.lastReportedTimeMillis = SystemClock.uptimeMillis(); 122 return state.lastReportedTimeMillis; 123 } 124 125 /** 126 * Loads an {@link Animation} object from a resource 127 * 128 * @param context Application context used to access resources 129 * @param id The resource id of the animation to load 130 * @return The animation object referenced by the specified id 131 * @throws NotFoundException when the animation cannot be loaded 132 */ loadAnimation(Context context, @AnimRes int id)133 public static Animation loadAnimation(Context context, @AnimRes int id) 134 throws NotFoundException { 135 136 XmlResourceParser parser = null; 137 try { 138 parser = context.getResources().getAnimation(id); 139 return createAnimationFromXml(context, parser); 140 } catch (XmlPullParserException ex) { 141 NotFoundException rnf = new NotFoundException("Can't load animation resource ID #0x" + 142 Integer.toHexString(id)); 143 rnf.initCause(ex); 144 throw rnf; 145 } catch (IOException ex) { 146 NotFoundException rnf = new NotFoundException("Can't load animation resource ID #0x" + 147 Integer.toHexString(id)); 148 rnf.initCause(ex); 149 throw rnf; 150 } finally { 151 if (parser != null) parser.close(); 152 } 153 } 154 createAnimationFromXml(Context c, XmlPullParser parser)155 private static Animation createAnimationFromXml(Context c, XmlPullParser parser) 156 throws XmlPullParserException, IOException { 157 158 return createAnimationFromXml(c, parser, null, Xml.asAttributeSet(parser)); 159 } 160 161 @UnsupportedAppUsage createAnimationFromXml(Context c, XmlPullParser parser, AnimationSet parent, AttributeSet attrs)162 private static Animation createAnimationFromXml(Context c, XmlPullParser parser, 163 AnimationSet parent, AttributeSet attrs) throws XmlPullParserException, IOException { 164 165 Animation anim = null; 166 167 // Make sure we are on a start tag. 168 int type; 169 int depth = parser.getDepth(); 170 171 while (((type=parser.next()) != XmlPullParser.END_TAG || parser.getDepth() > depth) 172 && type != XmlPullParser.END_DOCUMENT) { 173 174 if (type != XmlPullParser.START_TAG) { 175 continue; 176 } 177 178 String name = parser.getName(); 179 180 if (name.equals("set")) { 181 anim = new AnimationSet(c, attrs); 182 createAnimationFromXml(c, parser, (AnimationSet)anim, attrs); 183 } else if (name.equals("alpha")) { 184 anim = new AlphaAnimation(c, attrs); 185 } else if (name.equals("scale")) { 186 anim = new ScaleAnimation(c, attrs); 187 } else if (name.equals("rotate")) { 188 anim = new RotateAnimation(c, attrs); 189 } else if (name.equals("translate")) { 190 anim = new TranslateAnimation(c, attrs); 191 } else if (name.equals("cliprect")) { 192 anim = new ClipRectAnimation(c, attrs); 193 } else if (name.equals("extend")) { 194 anim = new ExtendAnimation(c, attrs); 195 } else { 196 throw new RuntimeException("Unknown animation name: " + parser.getName()); 197 } 198 199 if (parent != null) { 200 parent.addAnimation(anim); 201 } 202 } 203 204 return anim; 205 206 } 207 208 /** 209 * Loads a {@link LayoutAnimationController} object from a resource 210 * 211 * @param context Application context used to access resources 212 * @param id The resource id of the animation to load 213 * @return The animation controller object referenced by the specified id 214 * @throws NotFoundException when the layout animation controller cannot be loaded 215 */ loadLayoutAnimation(Context context, @AnimRes int id)216 public static LayoutAnimationController loadLayoutAnimation(Context context, @AnimRes int id) 217 throws NotFoundException { 218 219 XmlResourceParser parser = null; 220 try { 221 parser = context.getResources().getAnimation(id); 222 return createLayoutAnimationFromXml(context, parser); 223 } catch (XmlPullParserException ex) { 224 NotFoundException rnf = new NotFoundException("Can't load animation resource ID #0x" + 225 Integer.toHexString(id)); 226 rnf.initCause(ex); 227 throw rnf; 228 } catch (IOException ex) { 229 NotFoundException rnf = new NotFoundException("Can't load animation resource ID #0x" + 230 Integer.toHexString(id)); 231 rnf.initCause(ex); 232 throw rnf; 233 } finally { 234 if (parser != null) parser.close(); 235 } 236 } 237 createLayoutAnimationFromXml(Context c, XmlPullParser parser)238 private static LayoutAnimationController createLayoutAnimationFromXml(Context c, 239 XmlPullParser parser) throws XmlPullParserException, IOException { 240 241 return createLayoutAnimationFromXml(c, parser, Xml.asAttributeSet(parser)); 242 } 243 createLayoutAnimationFromXml(Context c, XmlPullParser parser, AttributeSet attrs)244 private static LayoutAnimationController createLayoutAnimationFromXml(Context c, 245 XmlPullParser parser, AttributeSet attrs) throws XmlPullParserException, IOException { 246 247 LayoutAnimationController controller = null; 248 249 int type; 250 int depth = parser.getDepth(); 251 252 while (((type = parser.next()) != XmlPullParser.END_TAG || parser.getDepth() > depth) 253 && type != XmlPullParser.END_DOCUMENT) { 254 255 if (type != XmlPullParser.START_TAG) { 256 continue; 257 } 258 259 String name = parser.getName(); 260 261 if ("layoutAnimation".equals(name)) { 262 controller = new LayoutAnimationController(c, attrs); 263 } else if ("gridLayoutAnimation".equals(name)) { 264 controller = new GridLayoutAnimationController(c, attrs); 265 } else { 266 throw new RuntimeException("Unknown layout animation name: " + name); 267 } 268 } 269 270 return controller; 271 } 272 273 /** 274 * Make an animation for objects becoming visible. Uses a slide and fade 275 * effect. 276 * 277 * @param c Context for loading resources 278 * @param fromLeft is the object to be animated coming from the left 279 * @return The new animation 280 */ makeInAnimation(Context c, boolean fromLeft)281 public static Animation makeInAnimation(Context c, boolean fromLeft) { 282 Animation a; 283 if (fromLeft) { 284 a = AnimationUtils.loadAnimation(c, com.android.internal.R.anim.slide_in_left); 285 } else { 286 a = AnimationUtils.loadAnimation(c, com.android.internal.R.anim.slide_in_right); 287 } 288 289 a.setInterpolator(new DecelerateInterpolator()); 290 a.setStartTime(currentAnimationTimeMillis()); 291 return a; 292 } 293 294 /** 295 * Make an animation for objects becoming invisible. Uses a slide and fade 296 * effect. 297 * 298 * @param c Context for loading resources 299 * @param toRight is the object to be animated exiting to the right 300 * @return The new animation 301 */ makeOutAnimation(Context c, boolean toRight)302 public static Animation makeOutAnimation(Context c, boolean toRight) { 303 Animation a; 304 if (toRight) { 305 a = AnimationUtils.loadAnimation(c, com.android.internal.R.anim.slide_out_right); 306 } else { 307 a = AnimationUtils.loadAnimation(c, com.android.internal.R.anim.slide_out_left); 308 } 309 310 a.setInterpolator(new AccelerateInterpolator()); 311 a.setStartTime(currentAnimationTimeMillis()); 312 return a; 313 } 314 315 316 /** 317 * Make an animation for objects becoming visible. Uses a slide up and fade 318 * effect. 319 * 320 * @param c Context for loading resources 321 * @return The new animation 322 */ makeInChildBottomAnimation(Context c)323 public static Animation makeInChildBottomAnimation(Context c) { 324 Animation a; 325 a = AnimationUtils.loadAnimation(c, com.android.internal.R.anim.slide_in_child_bottom); 326 a.setInterpolator(new AccelerateInterpolator()); 327 a.setStartTime(currentAnimationTimeMillis()); 328 return a; 329 } 330 331 /** 332 * Loads an {@link Interpolator} object from a resource 333 * 334 * @param context Application context used to access resources 335 * @param id The resource id of the animation to load 336 * @return The interpolator object referenced by the specified id 337 * @throws NotFoundException 338 */ loadInterpolator(Context context, @AnimRes @InterpolatorRes int id)339 public static Interpolator loadInterpolator(Context context, @AnimRes @InterpolatorRes int id) 340 throws NotFoundException { 341 XmlResourceParser parser = null; 342 try { 343 parser = context.getResources().getAnimation(id); 344 return createInterpolatorFromXml(context.getResources(), context.getTheme(), parser); 345 } catch (XmlPullParserException ex) { 346 NotFoundException rnf = new NotFoundException("Can't load animation resource ID #0x" + 347 Integer.toHexString(id)); 348 rnf.initCause(ex); 349 throw rnf; 350 } catch (IOException ex) { 351 NotFoundException rnf = new NotFoundException("Can't load animation resource ID #0x" + 352 Integer.toHexString(id)); 353 rnf.initCause(ex); 354 throw rnf; 355 } finally { 356 if (parser != null) parser.close(); 357 } 358 359 } 360 361 /** 362 * Loads an {@link Interpolator} object from a resource 363 * 364 * @param res The resources 365 * @param id The resource id of the animation to load 366 * @return The interpolator object referenced by the specified id 367 * @throws NotFoundException 368 * @hide 369 */ loadInterpolator(Resources res, Theme theme, int id)370 public static Interpolator loadInterpolator(Resources res, Theme theme, int id) throws NotFoundException { 371 XmlResourceParser parser = null; 372 try { 373 parser = res.getAnimation(id); 374 return createInterpolatorFromXml(res, theme, parser); 375 } catch (XmlPullParserException ex) { 376 NotFoundException rnf = new NotFoundException("Can't load animation resource ID #0x" + 377 Integer.toHexString(id)); 378 rnf.initCause(ex); 379 throw rnf; 380 } catch (IOException ex) { 381 NotFoundException rnf = new NotFoundException("Can't load animation resource ID #0x" + 382 Integer.toHexString(id)); 383 rnf.initCause(ex); 384 throw rnf; 385 } finally { 386 if (parser != null) 387 parser.close(); 388 } 389 390 } 391 createInterpolatorFromXml(Resources res, Theme theme, XmlPullParser parser)392 private static Interpolator createInterpolatorFromXml(Resources res, Theme theme, XmlPullParser parser) 393 throws XmlPullParserException, IOException { 394 395 BaseInterpolator interpolator = null; 396 397 // Make sure we are on a start tag. 398 int type; 399 int depth = parser.getDepth(); 400 401 while (((type = parser.next()) != XmlPullParser.END_TAG || parser.getDepth() > depth) 402 && type != XmlPullParser.END_DOCUMENT) { 403 404 if (type != XmlPullParser.START_TAG) { 405 continue; 406 } 407 408 AttributeSet attrs = Xml.asAttributeSet(parser); 409 410 String name = parser.getName(); 411 412 if (name.equals("linearInterpolator")) { 413 interpolator = new LinearInterpolator(); 414 } else if (name.equals("accelerateInterpolator")) { 415 interpolator = new AccelerateInterpolator(res, theme, attrs); 416 } else if (name.equals("decelerateInterpolator")) { 417 interpolator = new DecelerateInterpolator(res, theme, attrs); 418 } else if (name.equals("accelerateDecelerateInterpolator")) { 419 interpolator = new AccelerateDecelerateInterpolator(); 420 } else if (name.equals("cycleInterpolator")) { 421 interpolator = new CycleInterpolator(res, theme, attrs); 422 } else if (name.equals("anticipateInterpolator")) { 423 interpolator = new AnticipateInterpolator(res, theme, attrs); 424 } else if (name.equals("overshootInterpolator")) { 425 interpolator = new OvershootInterpolator(res, theme, attrs); 426 } else if (name.equals("anticipateOvershootInterpolator")) { 427 interpolator = new AnticipateOvershootInterpolator(res, theme, attrs); 428 } else if (name.equals("bounceInterpolator")) { 429 interpolator = new BounceInterpolator(); 430 } else if (name.equals("pathInterpolator")) { 431 interpolator = new PathInterpolator(res, theme, attrs); 432 } else { 433 throw new RuntimeException("Unknown interpolator name: " + parser.getName()); 434 } 435 } 436 return interpolator; 437 } 438 } 439