1282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski/* 2282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * Copyright (C) 2010 The Android Open Source Project 3282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * 4282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * Licensed under the Apache License, Version 2.0 (the "License"); 5282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * you may not use this file except in compliance with the License. 6282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * You may obtain a copy of the License at 7282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * 8282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * http://www.apache.org/licenses/LICENSE-2.0 9282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * 10282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * Unless required by applicable law or agreed to in writing, software 11282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * distributed under the License is distributed on an "AS IS" BASIS, 12282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * See the License for the specific language governing permissions and 14282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * limitations under the License. 15282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski */ 16282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski 17282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinskipackage android.animation; 18282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski 1929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perezimport com.android.layoutlib.bridge.Bridge; 20282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinskiimport com.android.layoutlib.bridge.impl.DelegateManager; 21282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinskiimport com.android.tools.layoutlib.annotations.LayoutlibDelegate; 22282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski 2329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perezimport java.lang.reflect.InvocationTargetException; 2429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perezimport java.lang.reflect.Method; 2529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perezimport java.util.Arrays; 2629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perezimport java.util.HashMap; 2729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perezimport java.util.Map; 2829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 29282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski/** 30282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * Delegate implementing the native methods of android.animation.PropertyValuesHolder 31282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * 32282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * Through the layoutlib_create tool, the original native methods of PropertyValuesHolder have been 33282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * replaced by calls to methods of the same name in this delegate class. 34282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * 35282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * Because it's a stateless class to start with, there's no need to keep a {@link DelegateManager} 36282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * around to map int to instance of the delegate. 37282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * 38282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * The main goal of this class' methods are to provide a native way to access setters and getters 3929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez * on some object. We override these methods to use reflection since the original reflection 4029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez * implementation of the PropertyValuesHolder won't be able to access protected methods. 41282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski * 42282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski */ 4329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez/*package*/ 4429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez@SuppressWarnings("unused") 4529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perezclass PropertyValuesHolder_Delegate { 4629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // This code is copied from android.animation.PropertyValuesHolder and must be kept in sync 4729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // We try several different types when searching for appropriate setter/getter functions. 4829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // The caller may have supplied values in a type that does not match the setter/getter 4929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // functions (such as the integers 0 and 1 to represent floating point values for alpha). 5029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // Also, the use of generics in constructors means that we end up with the Object versions 5129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // of primitive types (Float vs. float). But most likely, the setter/getter functions 5229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // will take primitive types instead. 5329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // So we supply an ordered array of other types to try before giving up. 5429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez private static Class[] FLOAT_VARIANTS = {float.class, Float.class, double.class, int.class, 5529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Double.class, Integer.class}; 5629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez private static Class[] INTEGER_VARIANTS = {int.class, Integer.class, float.class, double.class, 5729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Float.class, Double.class}; 5829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 5929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez private static final Object sMethodIndexLock = new Object(); 6029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez private static final Map<Long, Method> ID_TO_METHOD = new HashMap<Long, Method>(); 6129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez private static final Map<String, Long> METHOD_NAME_TO_ID = new HashMap<String, Long>(); 6229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez private static long sNextId = 1; 6329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 6429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez private static long registerMethod(Class<?> targetClass, String methodName, Class[] types, 6529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez int nArgs) { 6629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // Encode the number of arguments in the method name 67ada8c117b197dd61bd472399147dd18ff337a204Diego Perez String methodIndexName = String.format("%1$s.%2$s#%3$d", targetClass.getSimpleName(), 68ada8c117b197dd61bd472399147dd18ff337a204Diego Perez methodName, nArgs); 6929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez synchronized (sMethodIndexLock) { 7029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Long methodId = METHOD_NAME_TO_ID.get(methodIndexName); 7129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 7229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez if (methodId != null) { 7329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // The method was already registered 7429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez return methodId; 7529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 7629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 7729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Class[] args = new Class[nArgs]; 7829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Method method = null; 7929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez for (Class typeVariant : types) { 8029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez for (int i = 0; i < nArgs; i++) { 8129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez args[i] = typeVariant; 8229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 8329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez try { 8429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez method = targetClass.getDeclaredMethod(methodName, args); 8529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } catch (NoSuchMethodException ignore) { 8629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 8729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 8829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 8929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez if (method != null) { 9029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez methodId = sNextId++; 9129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez ID_TO_METHOD.put(methodId, method); 9229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez METHOD_NAME_TO_ID.put(methodIndexName, methodId); 9329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 9429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez return methodId; 9529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 9629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 9729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 9829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // Method not found 9929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez return 0; 10029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 10129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 10229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez private static void callMethod(Object target, long methodID, Object... args) { 10329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Method method = ID_TO_METHOD.get(methodID); 10429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez assert method != null; 10529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 10629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez try { 10729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez method.setAccessible(true); 10829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez method.invoke(target, args); 10929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } catch (IllegalAccessException e) { 11029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Bridge.getLog().error(null, "Unable to update property during animation", e, null); 11129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } catch (InvocationTargetException e) { 11229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Bridge.getLog().error(null, "Unable to update property during animation", e, null); 11329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 11429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 115282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski 116282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski @LayoutlibDelegate 117fbb35fb39eb74c6fa7ba6804faeaccb80483be14Ashok Bhat /*package*/ static long nGetIntMethod(Class<?> targetClass, String methodName) { 11829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez return nGetMultipleIntMethod(targetClass, methodName, 1); 119282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski } 120282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski 121282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski @LayoutlibDelegate 122fbb35fb39eb74c6fa7ba6804faeaccb80483be14Ashok Bhat /*package*/ static long nGetFloatMethod(Class<?> targetClass, String methodName) { 12329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez return nGetMultipleFloatMethod(targetClass, methodName, 1); 124282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski } 125282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski 126282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski @LayoutlibDelegate 1279be03c4e980d3058aeb3fd730da5f7d4a4a4f8a8Deepanshu Gupta /*package*/ static long nGetMultipleIntMethod(Class<?> targetClass, String methodName, 128e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta int numParams) { 12929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez return registerMethod(targetClass, methodName, INTEGER_VARIANTS, numParams); 130e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta } 131e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta 132e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta @LayoutlibDelegate 1339be03c4e980d3058aeb3fd730da5f7d4a4a4f8a8Deepanshu Gupta /*package*/ static long nGetMultipleFloatMethod(Class<?> targetClass, String methodName, 134e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta int numParams) { 13529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez return registerMethod(targetClass, methodName, FLOAT_VARIANTS, numParams); 136e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta } 137e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta 138e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta @LayoutlibDelegate 139fbb35fb39eb74c6fa7ba6804faeaccb80483be14Ashok Bhat /*package*/ static void nCallIntMethod(Object target, long methodID, int arg) { 14029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez callMethod(target, methodID, arg); 141282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski } 142282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski 143282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski @LayoutlibDelegate 144fbb35fb39eb74c6fa7ba6804faeaccb80483be14Ashok Bhat /*package*/ static void nCallFloatMethod(Object target, long methodID, float arg) { 14529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez callMethod(target, methodID, arg); 146282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski } 147e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta 148e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta @LayoutlibDelegate 149e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta /*package*/ static void nCallTwoIntMethod(Object target, long methodID, int arg1, 150e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta int arg2) { 15129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez callMethod(target, methodID, arg1, arg2); 152e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta } 153e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta 154e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta @LayoutlibDelegate 155e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta /*package*/ static void nCallFourIntMethod(Object target, long methodID, int arg1, 156e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta int arg2, int arg3, int arg4) { 15729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez callMethod(target, methodID, arg1, arg2, arg3, arg4); 158e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta } 159e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta 160e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta @LayoutlibDelegate 161e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta /*package*/ static void nCallMultipleIntMethod(Object target, long methodID, 162e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta int[] args) { 16329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez assert args != null; 16429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 16529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // Box parameters 16629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Object[] params = new Object[args.length]; 16729ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez for (int i = 0; i < args.length; i++) { 16829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez params[i] = args; 16929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 17029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez callMethod(target, methodID, params); 171e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta } 172e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta 173e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta @LayoutlibDelegate 174e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta /*package*/ static void nCallTwoFloatMethod(Object target, long methodID, float arg1, 175e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta float arg2) { 17629ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez callMethod(target, methodID, arg1, arg2); 177e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta } 178e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta 179e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta @LayoutlibDelegate 180e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta /*package*/ static void nCallFourFloatMethod(Object target, long methodID, float arg1, 181e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta float arg2, float arg3, float arg4) { 18229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez callMethod(target, methodID, arg1, arg2, arg3, arg4); 183e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta } 184e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta 185e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta @LayoutlibDelegate 186e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta /*package*/ static void nCallMultipleFloatMethod(Object target, long methodID, 187e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta float[] args) { 18829ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez assert args != null; 18929ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez 19029ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez // Box parameters 19129ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez Object[] params = new Object[args.length]; 19229ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez for (int i = 0; i < args.length; i++) { 19329ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez params[i] = args; 19429ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez } 19529ed07524ce0fc2e5950f5340d306247145d0efaDiego Perez callMethod(target, methodID, params); 196e05bb956ce429618fd4f971a9dc708b9313c59eaDeepanshu Gupta } 197282e181b58cf72b6ca770dc7ca5f91f135444502Adam Lesinski} 198