1 /* 2 * Copyright (C) 2016 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 androidx.leanback.widget; 18 19 import android.animation.ObjectAnimator; 20 import android.animation.PropertyValuesHolder; 21 import android.util.Property; 22 import android.view.animation.LinearInterpolator; 23 24 /** 25 * ParallaxTarget is responsible for updating the target through the {@link #update(float)} method 26 * or the {@link #directUpdate(Number)} method when {@link #isDirectMapping()} is true. 27 * When {@link #isDirectMapping()} is false, {@link ParallaxEffect} transforms the values of 28 * {@link Parallax}, which represents the current state of UI, into a float value between 0 and 1. 29 * That float value is passed into {@link #update(float)} method. 30 */ 31 public abstract class ParallaxTarget { 32 33 /** 34 * Implementation class is supposed to update target with the provided fraction 35 * (between 0 and 1). The fraction represents percentage of completed change (e.g. scroll) on 36 * target. Called only when {@link #isDirectMapping()} is false. 37 * 38 * @param fraction Fraction between 0 to 1. 39 * @see #isDirectMapping() 40 */ update(float fraction)41 public void update(float fraction) { 42 } 43 44 /** 45 * Returns true if the ParallaxTarget is directly mapping from source value, 46 * {@link #directUpdate(Number)} will be used to update value, otherwise update(fraction) will 47 * be called to update value. Default implementation returns false. 48 * 49 * @return True if direct mapping, false otherwise. 50 * @see #directUpdate(Number) 51 * @see #update(float) 52 */ isDirectMapping()53 public boolean isDirectMapping() { 54 return false; 55 } 56 57 /** 58 * Directly update the target using a float or int value. Called when {@link #isDirectMapping()} 59 * is true. 60 * 61 * @param value Either int or float value. 62 * @see #isDirectMapping() 63 */ directUpdate(Number value)64 public void directUpdate(Number value) { 65 } 66 67 /** 68 * PropertyValuesHolderTarget is an implementation of {@link ParallaxTarget} that uses 69 * {@link PropertyValuesHolder} to update the target object. 70 */ 71 public static final class PropertyValuesHolderTarget extends ParallaxTarget { 72 73 /** 74 * We simulate a parallax effect on target object using an ObjectAnimator. PSEUDO_DURATION 75 * is used on the ObjectAnimator. 76 */ 77 private static final long PSEUDO_DURATION = 1000000; 78 79 private final ObjectAnimator mAnimator; 80 private float mFraction; 81 PropertyValuesHolderTarget(Object targetObject, PropertyValuesHolder values)82 public PropertyValuesHolderTarget(Object targetObject, PropertyValuesHolder values) { 83 mAnimator = ObjectAnimator.ofPropertyValuesHolder(targetObject, values); 84 mAnimator.setInterpolator(new LinearInterpolator()); 85 mAnimator.setDuration(PSEUDO_DURATION); 86 } 87 88 @Override update(float fraction)89 public void update(float fraction) { 90 mFraction = fraction; 91 mAnimator.setCurrentPlayTime((long) (PSEUDO_DURATION * fraction)); 92 } 93 94 } 95 96 /** 97 * DirectPropertyTarget is to support direct mapping into either Integer Property or Float 98 * Property. App uses convenient method {@link ParallaxEffect#target(Object, Property)} to 99 * add a direct mapping. 100 * @param <T> Type of target object. 101 * @param <V> Type of value, either Integer or Float. 102 */ 103 public static final class DirectPropertyTarget<T extends Object, V extends Number> 104 extends ParallaxTarget { 105 106 Object mObject; 107 Property<T, V> mProperty; 108 109 /** 110 * @param targetObject Target object for perform Parallax 111 * @param property Target property, either an Integer Property or a Float Property. 112 */ DirectPropertyTarget(Object targetObject, Property<T, V> property)113 public DirectPropertyTarget(Object targetObject, Property<T, V> property) { 114 mObject = targetObject; 115 mProperty = property; 116 } 117 118 /** 119 * Returns true as DirectPropertyTarget receives a number to update Property in 120 * {@link #directUpdate(Number)}. 121 */ 122 @Override isDirectMapping()123 public boolean isDirectMapping() { 124 return true; 125 } 126 127 @Override directUpdate(Number value)128 public void directUpdate(Number value) { 129 mProperty.set((T) mObject, (V) value); 130 } 131 } 132 } 133