ViewDebug.java revision f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6
19066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/* 29066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project 39066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 49066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License"); 59066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * you may not use this file except in compliance with the License. 69066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * You may obtain a copy of the License at 79066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 89066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * http://www.apache.org/licenses/LICENSE-2.0 99066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Unless required by applicable law or agreed to in writing, software 119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS, 129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * See the License for the specific language governing permissions and 149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * limitations under the License. 159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpackage android.view; 189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 19e551dc76bcde0e114b06e876eeb1d5f9d367fddfRomain Guyimport android.util.Config; 209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.util.Log; 21c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Projectimport android.util.DisplayMetrics; 229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.content.res.Resources; 23105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Projectimport android.content.Context; 249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.graphics.Bitmap; 25c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Projectimport android.graphics.Canvas; 26223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guyimport android.graphics.Rect; 279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport android.os.Environment; 28c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Projectimport android.os.Debug; 29223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guyimport android.os.RemoteException; 309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 31223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guyimport java.io.ByteArrayOutputStream; 329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.File; 339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.BufferedWriter; 349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.FileWriter; 359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.IOException; 369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.FileOutputStream; 379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.DataOutputStream; 389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.OutputStreamWriter; 399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.BufferedOutputStream; 409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.io.OutputStream; 419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.List; 429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.LinkedList; 439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.ArrayList; 449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.HashMap; 459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.concurrent.CountDownLatch; 469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.util.concurrent.TimeUnit; 479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.lang.annotation.Target; 489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.lang.annotation.ElementType; 499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.lang.annotation.Retention; 509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.lang.annotation.RetentionPolicy; 519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.lang.reflect.Field; 529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.lang.reflect.Method; 539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectimport java.lang.reflect.InvocationTargetException; 54c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Projectimport java.lang.reflect.AccessibleObject; 559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project/** 579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Various debugging/tracing tools related to {@link View} and the view hierarchy. 589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Projectpublic class ViewDebug { 609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 6113922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * Log tag used to log errors related to the consistency of the view hierarchy. 6213922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * 6313922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * @hide 6413922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy */ 6513922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy public static final String CONSISTENCY_LOG_TAG = "ViewConsistency"; 6613922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy 6713922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy /** 6813922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * Flag indicating the consistency check should check layout-related properties. 6913922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * 7013922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * @hide 7113922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy */ 7213922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy public static final int CONSISTENCY_LAYOUT = 0x1; 7313922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy 7413922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy /** 7513922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * Flag indicating the consistency check should check drawing-related properties. 7613922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * 7713922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * @hide 7813922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy */ 7913922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy public static final int CONSISTENCY_DRAWING = 0x2; 8013922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy 8113922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy /** 829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Enables or disables view hierarchy tracing. Any invoker of 839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #trace(View, android.view.ViewDebug.HierarchyTraceType)} should first 849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * check that this value is set to true as not to affect performance. 859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static final boolean TRACE_HIERARCHY = false; 879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Enables or disables view recycler tracing. Any invoker of 909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #trace(View, android.view.ViewDebug.RecyclerTraceType, int[])} should first 919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * check that this value is set to true as not to affect performance. 929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static final boolean TRACE_RECYCLER = false; 94a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 96cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * Enables or disables motion events tracing. Any invoker of 97cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * {@link #trace(View, MotionEvent, MotionEventTraceType)} should first check 98cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * that this value is set to true as not to affect performance. 99cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 100cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @hide 101cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy */ 102cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy public static final boolean TRACE_MOTION_EVENTS = false; 103cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 104cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy /** 1059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The system property of dynamic switch for capturing view information 1069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * when it is set, we dump interested fields and methods for the view on focus 107a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy */ 1089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project static final String SYSTEM_PROPERTY_CAPTURE_VIEW = "debug.captureview"; 109a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 1109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The system property of dynamic switch for capturing event information 1129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * when it is set, we log key events, touch/motion and trackball events 113a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy */ 1149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project static final String SYSTEM_PROPERTY_CAPTURE_EVENT = "debug.captureevent"; 115c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 11713922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * Profiles drawing times in the events log. 11813922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * 11913922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * @hide 12013922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy */ 12113922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy @Debug.DebugProperty 12213922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy public static boolean profileDrawing = false; 12313922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy 12413922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy /** 12513922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * Profiles layout times in the events log. 12613922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * 12713922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * @hide 12813922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy */ 12913922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy @Debug.DebugProperty 13013922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy public static boolean profileLayout = false; 13113922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy 13213922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy /** 13313922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * Profiles real fps (times between draws) and displays the result. 13413922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * 13513922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * @hide 13613922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy */ 13713922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy @Debug.DebugProperty 13813922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy public static boolean showFps = false; 13913922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy 14013922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy /** 14113922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * <p>Enables or disables views consistency check. Even when this property is enabled, 14213922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * view consistency checks happen only if {@link android.util.Config#DEBUG} is set 14313922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * to true. The value of this property can be configured externally in one of the 14413922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * following files:</p> 14513922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * <ul> 14613922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * <li>/system/debug.prop</li> 14713922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * <li>/debug.prop</li> 14813922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * <li>/data/debug.prop</li> 14913922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * </ul> 15013922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy * @hide 15113922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy */ 15213922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy @Debug.DebugProperty 15313922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy public static boolean consistencyCheckEnabled = false; 15413922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy 15513922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy static { 156e551dc76bcde0e114b06e876eeb1d5f9d367fddfRomain Guy if (Config.DEBUG) { 157e551dc76bcde0e114b06e876eeb1d5f9d367fddfRomain Guy Debug.setFieldsOn(ViewDebug.class, true); 158e551dc76bcde0e114b06e876eeb1d5f9d367fddfRomain Guy } 15913922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy } 16013922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy 16113922e03037d71a538f24ddf61c0b61bb4eb5af0Romain Guy /** 1629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * This annotation can be used to mark fields and methods to be dumped by 1639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * the view server. Only non-void methods with no arguments can be annotated 1649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * by this annotation. 1659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project @Target({ ElementType.FIELD, ElementType.METHOD }) 1679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project @Retention(RetentionPolicy.RUNTIME) 1689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public @interface ExportedProperty { 1699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * When resolveId is true, and if the annotated field/method return value 1719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * is an int, the value is converted to an Android's resource name. 1729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 1739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return true if the property's value must be transformed into an Android 1749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * resource name, false otherwise 1759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project boolean resolveId() default false; 1779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * A mapping can be defined to map int values to specific strings. For 1809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * instance, View.getVisibility() returns 0, 4 or 8. However, these values 1819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * actually mean VISIBLE, INVISIBLE and GONE. A mapping can be used to see 1829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * these human readable values: 1839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 1849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <pre> 1859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @ViewDebug.ExportedProperty(mapping = { 1869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @ViewDebug.IntToString(from = 0, to = "VISIBLE"), 1879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @ViewDebug.IntToString(from = 4, to = "INVISIBLE"), 1889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @ViewDebug.IntToString(from = 8, to = "GONE") 1899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * }) 1909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * public int getVisibility() { ... 1919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <pre> 1929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 1939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return An array of int to String mappings 1949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 1959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see android.view.ViewDebug.IntToString 1969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project IntToString[] mapping() default { }; 1989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 200c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * A mapping can be defined to map array indices to specific strings. 201c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * A mapping can be used to see human readable values for the indices 202c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * of an array: 203c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * 204c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * <pre> 205809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @ViewDebug.ExportedProperty(indexMapping = { 206c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * @ViewDebug.IntToString(from = 0, to = "INVALID"), 207c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * @ViewDebug.IntToString(from = 1, to = "FIRST"), 208c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * @ViewDebug.IntToString(from = 2, to = "SECOND") 209c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * }) 210c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * private int[] mElements; 211c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * <pre> 212c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * 213c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * @return An array of int to String mappings 214c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * 215c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * @see android.view.ViewDebug.IntToString 216c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project * @see #mapping() 217c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project */ 218c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project IntToString[] indexMapping() default { }; 219c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 220c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project /** 221809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * A flags mapping can be defined to map flags encoded in an integer to 222809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * specific strings. A mapping can be used to see human readable values 223809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * for the flags of an integer: 224809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * 225809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * <pre> 226809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @ViewDebug.ExportedProperty(flagMapping = { 227809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @ViewDebug.FlagToString(mask = ENABLED_MASK, equals = ENABLED, name = "ENABLED"), 228809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @ViewDebug.FlagToString(mask = ENABLED_MASK, equals = DISABLED, name = "DISABLED"), 229809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * }) 230809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * private int mFlags; 231809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * <pre> 232809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * 233809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * A specified String is output when the following is true: 234a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy * 235809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @return An array of int to String mappings 236809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy */ 237809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy FlagToString[] flagMapping() default { }; 238809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 239809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy /** 2409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * When deep export is turned on, this property is not dumped. Instead, the 2419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * properties contained in this property are dumped. Each child property 2429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * is prefixed with the name of this property. 2439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return true if the properties of this property should be dumped 2459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 246a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy * @see #prefix() 2479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 2489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project boolean deepExport() default false; 2499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 2519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The prefix to use on child properties when deep export is enabled 2529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return a prefix as a String 2549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see #deepExport() 2569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 2579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String prefix() default ""; 2589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 2599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 2619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Defines a mapping from an int value to a String. Such a mapping can be used 2629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * in a @ExportedProperty to provide more meaningful values to the end user. 2639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see android.view.ViewDebug.ExportedProperty 2659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 2669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project @Target({ ElementType.TYPE }) 2679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project @Retention(RetentionPolicy.RUNTIME) 2689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public @interface IntToString { 2699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 2709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The original int value to map to a String. 2719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return An arbitrary int value. 2739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 2749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int from(); 2759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 2769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 2779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * The String to use in place of the original int value. 2789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 2799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return An arbitrary non-null String. 2809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 2819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String to(); 2829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 283809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 284809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy /** 285809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * Defines a mapping from an flag to a String. Such a mapping can be used 286809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * in a @ExportedProperty to provide more meaningful values to the end user. 287809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * 288809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @see android.view.ViewDebug.ExportedProperty 289809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy */ 290809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy @Target({ ElementType.TYPE }) 291809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy @Retention(RetentionPolicy.RUNTIME) 292809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy public @interface FlagToString { 293809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy /** 294809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * The mask to apply to the original value. 295809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * 296809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @return An arbitrary int value. 297809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy */ 298809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy int mask(); 299809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 300809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy /** 301809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * The value to compare to the result of: 302809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * <code>original value & {@link #mask()}</code>. 303809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * 304809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @return An arbitrary value. 305809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy */ 306809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy int equals(); 307809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 308809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy /** 309809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * The String to use in place of the original int value. 310809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * 311809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * @return An arbitrary non-null String. 312809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy */ 313809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy String name(); 314809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 315809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy /** 316809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * Indicates whether to output the flag when the test is true, 317809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy * or false. Defaults to true. 318809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy */ 319809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy boolean outputIf() default true; 320809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy } 321809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 3229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 3239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * This annotation can be used to mark fields and methods to be dumped when 3249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * the view is captured. Methods with this annotation must have no arguments 325f8a7ceaef2e7d5cd530c9426bde91b6fa9a40b75Andy Stadler * and must return a valid type of data. 3269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 3279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project @Target({ ElementType.FIELD, ElementType.METHOD }) 3289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project @Retention(RetentionPolicy.RUNTIME) 3299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public @interface CapturedViewProperty { 3309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 331a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy * When retrieveReturn is true, we need to retrieve second level methods 3329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * e.g., we need myView.getFirstLevelMethod().getSecondLevelMethod() 333a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy * we will set retrieveReturn = true on the annotation of 3349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * myView.getFirstLevelMethod() 335a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy * @return true if we need the second level methods 3369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 337a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy boolean retrieveReturn() default false; 3389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 339a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 3409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static HashMap<Class<?>, Method[]> mCapturedViewMethodsForClasses = null; 3419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static HashMap<Class<?>, Field[]> mCapturedViewFieldsForClasses = null; 3429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Maximum delay in ms after which we stop trying to capture a View's drawing 3449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final int CAPTURE_TIMEOUT = 4000; 3459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final String REMOTE_COMMAND_CAPTURE = "CAPTURE"; 3479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final String REMOTE_COMMAND_DUMP = "DUMP"; 3489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final String REMOTE_COMMAND_INVALIDATE = "INVALIDATE"; 3499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static final String REMOTE_COMMAND_REQUEST_LAYOUT = "REQUEST_LAYOUT"; 350c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project private static final String REMOTE_PROFILE = "PROFILE"; 351223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy private static final String REMOTE_COMMAND_CAPTURE_LAYERS = "CAPTURE_LAYERS"; 3529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static HashMap<Class<?>, Field[]> sFieldsForClasses; 3549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static HashMap<Class<?>, Method[]> sMethodsForClasses; 355c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project private static HashMap<AccessibleObject, ExportedProperty> sAnnotations; 356c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 3579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 3589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Defines the type of hierarhcy trace to output to the hierarchy traces file. 3599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 3609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public enum HierarchyTraceType { 3619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project INVALIDATE, 3629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project INVALIDATE_CHILD, 3639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project INVALIDATE_CHILD_IN_PARENT, 3649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project REQUEST_LAYOUT, 3659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ON_LAYOUT, 3669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ON_MEASURE, 3679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project DRAW, 3689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project BUILD_CACHE 3699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static BufferedWriter sHierarchyTraces; 3729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static ViewRoot sHierarhcyRoot; 3739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static String sHierarchyTracePrefix; 3749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 3769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Defines the type of recycler trace to output to the recycler traces file. 3779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 3789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public enum RecyclerTraceType { 3799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project NEW_VIEW, 3809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project BIND_VIEW, 3819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project RECYCLE_FROM_ACTIVE_HEAP, 3829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project RECYCLE_FROM_SCRAP_HEAP, 3839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project MOVE_TO_SCRAP_HEAP, 3849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project MOVE_FROM_ACTIVE_TO_SCRAP_HEAP 3859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static class RecyclerTrace { 3889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int view; 3899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public RecyclerTraceType type; 3909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int position; 3919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public int indexOnScreen; 3929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 3939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static View sRecyclerOwnerView; 3959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static List<View> sRecyclerViews; 3969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static List<RecyclerTrace> sRecyclerTraces; 3979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static String sRecyclerTracePrefix; 3989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 3999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 400cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * Defines the type of motion events trace to output to the motion events traces file. 401cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 402cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @hide 403cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy */ 404cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy public enum MotionEventTraceType { 405cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy DISPATCH, 406cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy ON_INTERCEPT, 407cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy ON_TOUCH 408cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 409cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 410cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy private static BufferedWriter sMotionEventTraces; 411cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy private static ViewRoot sMotionEventRoot; 412cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy private static String sMotionEventTracePrefix; 413cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 414cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy /** 4159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Returns the number of instanciated Views. 4169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return The number of Views instanciated in the current process. 4189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @hide 4209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 4219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static long getViewInstanceCount() { 4229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return View.sInstanceCount; 4239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 4269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Returns the number of instanciated ViewRoots. 4279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @return The number of ViewRoots instanciated in the current process. 4299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @hide 4319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 4329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static long getViewRootInstanceCount() { 4339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return ViewRoot.getInstanceCount(); 434a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy } 4359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 4379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Outputs a trace to the currently opened recycler traces. The trace records the type of 4389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * recycler action performed on the supplied view as well as a number of parameters. 4399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param view the view to trace 4419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param type the type of the trace 4429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param parameters parameters depending on the type of the trace 4439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 4449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static void trace(View view, RecyclerTraceType type, int... parameters) { 4459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sRecyclerOwnerView == null || sRecyclerViews == null) { 4469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 4479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!sRecyclerViews.contains(view)) { 4509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerViews.add(view); 4519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final int index = sRecyclerViews.indexOf(view); 4549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project RecyclerTrace trace = new RecyclerTrace(); 4569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project trace.view = index; 4579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project trace.type = type; 4589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project trace.position = parameters[0]; 4599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project trace.indexOnScreen = parameters[1]; 4609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerTraces.add(trace); 4629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 4659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Starts tracing the view recycler of the specified view. The trace is identified by a prefix, 4669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * used to build the traces files names: <code>/EXTERNAL/view-recycler/PREFIX.traces</code> and 4679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <code>/EXTERNAL/view-recycler/PREFIX.recycler</code>. 4689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Only one view recycler can be traced at the same time. After calling this method, any 4709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * other invocation will result in a <code>IllegalStateException</code> unless 4719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #stopRecyclerTracing()} is invoked before. 4729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Traces files are created only after {@link #stopRecyclerTracing()} is invoked. 4749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * This method will return immediately if TRACE_RECYCLER is false. 4769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param prefix the traces files name prefix 4789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param view the view whose recycler must be traced 4799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 4809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see #stopRecyclerTracing() 4819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see #trace(View, android.view.ViewDebug.RecyclerTraceType, int[]) 4829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 4839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static void startRecyclerTracing(String prefix, View view) { 4849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //noinspection PointlessBooleanExpression,ConstantConditions 4859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!TRACE_RECYCLER) { 4869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 4879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sRecyclerOwnerView != null) { 4909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IllegalStateException("You must call stopRecyclerTracing() before running" + 4919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project " a new trace!"); 4929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 4949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerTracePrefix = prefix; 4959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerOwnerView = view; 4969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerViews = new ArrayList<View>(); 4979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerTraces = new LinkedList<RecyclerTrace>(); 4989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 4999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 5019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Stops the current view recycer tracing. 5029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 5039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Calling this method creates the file <code>/EXTERNAL/view-recycler/PREFIX.traces</code> 5049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * containing all the traces (or method calls) relative to the specified view's recycler. 5059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 5069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Calling this method creates the file <code>/EXTERNAL/view-recycler/PREFIX.recycler</code> 5079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * containing all of the views used by the recycler of the view supplied to 5089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #startRecyclerTracing(String, View)}. 5099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 5109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * This method will return immediately if TRACE_RECYCLER is false. 5119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 5129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see #startRecyclerTracing(String, View) 5139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see #trace(View, android.view.ViewDebug.RecyclerTraceType, int[]) 5149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 5159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static void stopRecyclerTracing() { 5169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //noinspection PointlessBooleanExpression,ConstantConditions 5179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!TRACE_RECYCLER) { 5189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 5199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sRecyclerOwnerView == null || sRecyclerViews == null) { 5229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IllegalStateException("You must call startRecyclerTracing() before" + 5239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project " stopRecyclerTracing()!"); 5249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project File recyclerDump = new File(Environment.getExternalStorageDirectory(), "view-recycler/"); 527c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project //noinspection ResultOfMethodCallIgnored 5289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project recyclerDump.mkdirs(); 5299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project recyclerDump = new File(recyclerDump, sRecyclerTracePrefix + ".recycler"); 5319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 5329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final BufferedWriter out = new BufferedWriter(new FileWriter(recyclerDump), 8 * 1024); 5339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (View view : sRecyclerViews) { 5359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final String name = view.getClass().getName(); 5369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write(name); 5379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.newLine(); 5389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.close(); 5419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 5429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.e("View", "Could not dump recycler content"); 5439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 5449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project recyclerDump = new File(Environment.getExternalStorageDirectory(), "view-recycler/"); 5479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project recyclerDump = new File(recyclerDump, sRecyclerTracePrefix + ".traces"); 5489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 5491afd1c90ebe789b8d3a137004127a50d2db7e3b5Dianne Hackborn if (recyclerDump.exists()) { 5501afd1c90ebe789b8d3a137004127a50d2db7e3b5Dianne Hackborn recyclerDump.delete(); 5511afd1c90ebe789b8d3a137004127a50d2db7e3b5Dianne Hackborn } 5529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final FileOutputStream file = new FileOutputStream(recyclerDump); 5539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final DataOutputStream out = new DataOutputStream(file); 5549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (RecyclerTrace trace : sRecyclerTraces) { 5569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.writeInt(trace.view); 5579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.writeInt(trace.type.ordinal()); 5589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.writeInt(trace.position); 5599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.writeInt(trace.indexOnScreen); 5609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.flush(); 5619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.close(); 5649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 5659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.e("View", "Could not dump recycler traces"); 5669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 5679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerViews.clear(); 5709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerViews = null; 5719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerTraces.clear(); 5739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerTraces = null; 5749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sRecyclerOwnerView = null; 5769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 5799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Outputs a trace to the currently opened traces file. The trace contains the class name 5809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * and instance's hashcode of the specified view as well as the supplied trace type. 5819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 5829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param view the view to trace 5839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param type the type of the trace 5849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 5859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static void trace(View view, HierarchyTraceType type) { 5869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sHierarchyTraces == null) { 5879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 5889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 5899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 5909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 5919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces.write(type.name()); 5929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces.write(' '); 5939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces.write(view.getClass().getName()); 5949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces.write('@'); 5959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces.write(Integer.toHexString(view.hashCode())); 5969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces.newLine(); 5979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 5989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.w("View", "Error while dumping trace of type " + type + " for view " + view); 5999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 6039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Starts tracing the view hierarchy of the specified view. The trace is identified by a prefix, 6049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * used to build the traces files names: <code>/EXTERNAL/view-hierarchy/PREFIX.traces</code> and 6059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <code>/EXTERNAL/view-hierarchy/PREFIX.tree</code>. 6069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 6079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Only one view hierarchy can be traced at the same time. After calling this method, any 6089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * other invocation will result in a <code>IllegalStateException</code> unless 6099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #stopHierarchyTracing()} is invoked before. 6109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 6119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Calling this method creates the file <code>/EXTERNAL/view-hierarchy/PREFIX.traces</code> 6129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * containing all the traces (or method calls) relative to the specified view's hierarchy. 6139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 6149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * This method will return immediately if TRACE_HIERARCHY is false. 6159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 6169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param prefix the traces files name prefix 6179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param view the view whose hierarchy must be traced 6189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 6199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see #stopHierarchyTracing() 6209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see #trace(View, android.view.ViewDebug.HierarchyTraceType) 6219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 6229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static void startHierarchyTracing(String prefix, View view) { 6239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //noinspection PointlessBooleanExpression,ConstantConditions 6249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!TRACE_HIERARCHY) { 6259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sHierarhcyRoot != null) { 6299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IllegalStateException("You must call stopHierarchyTracing() before running" + 6309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project " a new trace!"); 6319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project File hierarchyDump = new File(Environment.getExternalStorageDirectory(), "view-hierarchy/"); 634c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project //noinspection ResultOfMethodCallIgnored 6359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project hierarchyDump.mkdirs(); 6369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project hierarchyDump = new File(hierarchyDump, prefix + ".traces"); 6389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTracePrefix = prefix; 6399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 6419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces = new BufferedWriter(new FileWriter(hierarchyDump), 8 * 1024); 6429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 6439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.e("View", "Could not dump view hierarchy"); 6449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarhcyRoot = (ViewRoot) view.getRootView().getParent(); 6489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 6519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Stops the current view hierarchy tracing. This method closes the file 6529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * <code>/EXTERNAL/view-hierarchy/PREFIX.traces</code>. 6539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 6549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * Calling this method creates the file <code>/EXTERNAL/view-hierarchy/PREFIX.tree</code> 6559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * containing the view hierarchy of the view supplied to 6569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * {@link #startHierarchyTracing(String, View)}. 6579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 6589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * This method will return immediately if TRACE_HIERARCHY is false. 6599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * 660a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy * @see #startHierarchyTracing(String, View) 6619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @see #trace(View, android.view.ViewDebug.HierarchyTraceType) 6629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 6639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public static void stopHierarchyTracing() { 6649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //noinspection PointlessBooleanExpression,ConstantConditions 6659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!TRACE_HIERARCHY) { 6669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sHierarhcyRoot == null || sHierarchyTraces == null) { 6709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throw new IllegalStateException("You must call startHierarchyTracing() before" + 6719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project " stopHierarchyTracing()!"); 6729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 6759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces.close(); 6769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 6779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.e("View", "Could not write view traces"); 6789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarchyTraces = null; 6809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project File hierarchyDump = new File(Environment.getExternalStorageDirectory(), "view-hierarchy/"); 682c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project //noinspection ResultOfMethodCallIgnored 6839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project hierarchyDump.mkdirs(); 6849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project hierarchyDump = new File(hierarchyDump, sHierarchyTracePrefix + ".tree"); 6859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project BufferedWriter out; 6879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 6889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out = new BufferedWriter(new FileWriter(hierarchyDump), 8 * 1024); 6899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 6909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.e("View", "Could not dump view hierarchy"); 6919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 6929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 6939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 6949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project View view = sHierarhcyRoot.getView(); 6959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (view instanceof ViewGroup) { 6969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ViewGroup group = (ViewGroup) view; 6979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project dumpViewHierarchy(group, out, 0); 6989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 6999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.close(); 7009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 7019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.e("View", "Could not dump view hierarchy"); 7029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 7049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 7059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sHierarhcyRoot = null; 7069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 707a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 708cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy /** 709cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * Outputs a trace to the currently opened traces file. The trace contains the class name 710cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * and instance's hashcode of the specified view as well as the supplied trace type. 711cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 712cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @param view the view to trace 713cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @param event the event of the trace 714cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @param type the type of the trace 715cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 716cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @hide 717cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy */ 718cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy public static void trace(View view, MotionEvent event, MotionEventTraceType type) { 719cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy if (sMotionEventTraces == null) { 720cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy return; 721cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 722cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 723cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy try { 724cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces.write(type.name()); 725cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces.write(' '); 726cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces.write(event.getAction()); 727cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces.write(' '); 728cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces.write(view.getClass().getName()); 729cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces.write('@'); 730cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces.write(Integer.toHexString(view.hashCode())); 731cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sHierarchyTraces.newLine(); 732cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } catch (IOException e) { 733cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy Log.w("View", "Error while dumping trace of event " + event + " for view " + view); 734cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 735cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 736cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 737cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy /** 738cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * Starts tracing the motion events for the hierarchy of the specificy view. 739cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * The trace is identified by a prefix, used to build the traces files names: 740cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * <code>/EXTERNAL/motion-events/PREFIX.traces</code> and 741cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * <code>/EXTERNAL/motion-events/PREFIX.tree</code>. 742cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 743cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * Only one view hierarchy can be traced at the same time. After calling this method, any 744cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * other invocation will result in a <code>IllegalStateException</code> unless 745cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * {@link #stopMotionEventTracing()} is invoked before. 746cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 747cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * Calling this method creates the file <code>/EXTERNAL/motion-events/PREFIX.traces</code> 748cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * containing all the traces (or method calls) relative to the specified view's hierarchy. 749cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 750cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * This method will return immediately if TRACE_HIERARCHY is false. 751cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 752cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @param prefix the traces files name prefix 753cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @param view the view whose hierarchy must be traced 754cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 755cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @see #stopMotionEventTracing() 756cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @see #trace(View, MotionEvent, android.view.ViewDebug.MotionEventTraceType) 757cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 758cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @hide 759cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy */ 760cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy public static void startMotionEventTracing(String prefix, View view) { 761cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy //noinspection PointlessBooleanExpression,ConstantConditions 762cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy if (!TRACE_MOTION_EVENTS) { 763cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy return; 764cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 765cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 766cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy if (sMotionEventRoot != null) { 767cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy throw new IllegalStateException("You must call stopMotionEventTracing() before running" + 768cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy " a new trace!"); 769cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 770cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 771cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy File hierarchyDump = new File(Environment.getExternalStorageDirectory(), "motion-events/"); 772cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy //noinspection ResultOfMethodCallIgnored 773cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy hierarchyDump.mkdirs(); 774cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 775cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy hierarchyDump = new File(hierarchyDump, prefix + ".traces"); 776cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTracePrefix = prefix; 777cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 778cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy try { 779cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces = new BufferedWriter(new FileWriter(hierarchyDump), 32 * 1024); 780cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } catch (IOException e) { 781cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy Log.e("View", "Could not dump view hierarchy"); 782cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy return; 783cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 784cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 785cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventRoot = (ViewRoot) view.getRootView().getParent(); 786cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 787cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 788cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy /** 789cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * Stops the current motion events tracing. This method closes the file 790cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * <code>/EXTERNAL/motion-events/PREFIX.traces</code>. 791cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 792cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * Calling this method creates the file <code>/EXTERNAL/motion-events/PREFIX.tree</code> 793cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * containing the view hierarchy of the view supplied to 794cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * {@link #startMotionEventTracing(String, View)}. 795cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 796cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * This method will return immediately if TRACE_HIERARCHY is false. 797cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 798cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @see #startMotionEventTracing(String, View) 799cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @see #trace(View, MotionEvent, android.view.ViewDebug.MotionEventTraceType) 800cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * 801cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy * @hide 802cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy */ 803cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy public static void stopMotionEventTracing() { 804cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy //noinspection PointlessBooleanExpression,ConstantConditions 805cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy if (!TRACE_MOTION_EVENTS) { 806cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy return; 807cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 808cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 809cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy if (sMotionEventRoot == null || sMotionEventTraces == null) { 810cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy throw new IllegalStateException("You must call startMotionEventTracing() before" + 811cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy " stopMotionEventTracing()!"); 812cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 813cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 814cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy try { 815cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces.close(); 816cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } catch (IOException e) { 817cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy Log.e("View", "Could not write view traces"); 818cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 819cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sMotionEventTraces = null; 820cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 821cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy File hierarchyDump = new File(Environment.getExternalStorageDirectory(), "motion-events/"); 822cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy //noinspection ResultOfMethodCallIgnored 823cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy hierarchyDump.mkdirs(); 824cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy hierarchyDump = new File(hierarchyDump, sMotionEventTracePrefix + ".tree"); 825cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 826cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy BufferedWriter out; 827cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy try { 828cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy out = new BufferedWriter(new FileWriter(hierarchyDump), 8 * 1024); 829cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } catch (IOException e) { 830cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy Log.e("View", "Could not dump view hierarchy"); 831cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy return; 832cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 833cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 834cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy View view = sMotionEventRoot.getView(); 835cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy if (view instanceof ViewGroup) { 836cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy ViewGroup group = (ViewGroup) view; 837cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy dumpViewHierarchy(group, out, 0); 838cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy try { 839cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy out.close(); 840cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } catch (IOException e) { 841cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy Log.e("View", "Could not dump view hierarchy"); 842cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 843cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 844cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 845cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy sHierarhcyRoot = null; 846cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy } 847cf635ae7c65de34f62a41cd330f25f4ee9ddc95bRomain Guy 8489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project static void dispatchCommand(View view, String command, String parameters, 8499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project OutputStream clientStream) throws IOException { 8509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // Paranoid but safe... 8529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project view = view.getRootView(); 8539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (REMOTE_COMMAND_DUMP.equalsIgnoreCase(command)) { 8559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project dump(view, clientStream); 856223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } else if (REMOTE_COMMAND_CAPTURE_LAYERS.equalsIgnoreCase(command)) { 857223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy captureLayers(view, new DataOutputStream(clientStream)); 8589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 8599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final String[] params = parameters.split(" "); 8609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (REMOTE_COMMAND_CAPTURE.equalsIgnoreCase(command)) { 8619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project capture(view, clientStream, params[0]); 8629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (REMOTE_COMMAND_INVALIDATE.equalsIgnoreCase(command)) { 8639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project invalidate(view, params[0]); 8649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (REMOTE_COMMAND_REQUEST_LAYOUT.equalsIgnoreCase(command)) { 8659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project requestLayout(view, params[0]); 866c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } else if (REMOTE_PROFILE.equalsIgnoreCase(command)) { 867c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project profile(view, clientStream, params[0]); 8689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 872c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project private static View findView(View root, String parameter) { 873c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project // Look by type/hashcode 874c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (parameter.indexOf('@') != -1) { 875c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final String[] ids = parameter.split("@"); 876c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final String className = ids[0]; 877236092a36216c79507ec19eb207831810caced19Romain Guy final int hashCode = (int) Long.parseLong(ids[1], 16); 8789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 879c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project View view = root.getRootView(); 880c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (view instanceof ViewGroup) { 881c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project return findView((ViewGroup) view, className, hashCode); 882c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 883c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } else { 884c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project // Look by id 885c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final int id = root.getResources().getIdentifier(parameter, null, null); 886c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project return root.getRootView().findViewById(id); 8879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return null; 8909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static void invalidate(View root, String parameter) { 893c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final View view = findView(root, parameter); 8949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (view != null) { 8959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project view.postInvalidate(); 8969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 8989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 8999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static void requestLayout(View root, String parameter) { 900c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final View view = findView(root, parameter); 9019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (view != null) { 9029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project root.post(new Runnable() { 9039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void run() { 9049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project view.requestLayout(); 9059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project }); 9079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 9099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 910c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project private static void profile(View root, OutputStream clientStream, String parameter) 9119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project throws IOException { 9129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 913c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final View view = findView(root, parameter); 914c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project BufferedWriter out = null; 915c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project try { 916c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out = new BufferedWriter(new OutputStreamWriter(clientStream), 32 * 1024); 917c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 918c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (view != null) { 919f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev profileViewAndChildren(view, out); 920f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } else { 921f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.write("-1 -1 -1"); 922f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.newLine(); 923f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 924f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.write("DONE."); 925f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.newLine(); 926f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } catch (Exception e) { 927f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev android.util.Log.w("View", "Problem profiling the view:", e); 928f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } finally { 929f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev if (out != null) { 930f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.close(); 931f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 932f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 933f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 934c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 935f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev private static void profileViewAndChildren(final View view, BufferedWriter out) 936f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev throws IOException { 937f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev final long durationMeasure = profileViewOperation(view, new ViewOperation<Void>() { 938f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public Void[] pre() { 939f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev forceLayout(view); 940f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev return null; 941f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 942c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 943f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev private void forceLayout(View view) { 944f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev view.forceLayout(); 945f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev if (view instanceof ViewGroup) { 946f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev ViewGroup group = (ViewGroup) view; 947f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev final int count = group.getChildCount(); 948f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev for (int i = 0; i < count; i++) { 949f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev forceLayout(group.getChildAt(i)); 950c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 951f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 952f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 953c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 954f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public void run(Void... data) { 955f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev view.measure(view.mOldWidthMeasureSpec, view.mOldHeightMeasureSpec); 956f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 957c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 958f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public void post(Void... data) { 959f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 960f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev }); 961c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 962f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev final long durationLayout = profileViewOperation(view, new ViewOperation<Void>() { 963f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public Void[] pre() { 964f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev return null; 965f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 966c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 967f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public void run(Void... data) { 968f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev view.layout(view.mLeft, view.mTop, view.mRight, view.mBottom); 969f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 970c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 971f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public void post(Void... data) { 972f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 973f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev }); 974c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 975f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev final long durationDraw = profileViewOperation(view, new ViewOperation<Object>() { 976f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public Object[] pre() { 977f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev final DisplayMetrics metrics = view.getResources().getDisplayMetrics(); 978f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev final Bitmap bitmap = 979f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev Bitmap.createBitmap(metrics.widthPixels, metrics.heightPixels, 980f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev Bitmap.Config.RGB_565); 981f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev final Canvas canvas = new Canvas(bitmap); 982f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev return new Object[] { 983f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev bitmap, canvas 984f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev }; 985f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 986c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 987f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public void run(Object... data) { 988f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev view.draw((Canvas) data[1]); 989c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 990f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev 991f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev public void post(Object... data) { 992f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev ((Bitmap) data[0]).recycle(); 993f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev } 994f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev }); 995f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev 996f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.write(String.valueOf(durationMeasure)); 997f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.write(' '); 998f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.write(String.valueOf(durationLayout)); 999f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.write(' '); 1000f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.write(String.valueOf(durationDraw)); 1001f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev out.newLine(); 1002f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev if (view instanceof ViewGroup) { 1003f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev ViewGroup group = (ViewGroup) view; 1004f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev final int count = group.getChildCount(); 1005f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev for (int i = 0; i < count; i++) { 1006f8e1219cf8992a22d6c48c2c70d4fbbccc2494f6Konstantin Lopyrev profileViewAndChildren(group.getChildAt(i), out); 1007c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1008c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1009c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1010c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1011c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project interface ViewOperation<T> { 1012c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project T[] pre(); 1013c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project void run(T... data); 1014c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project void post(T... data); 1015c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1016c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1017c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project private static <T> long profileViewOperation(View view, final ViewOperation<T> operation) { 10189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final CountDownLatch latch = new CountDownLatch(1); 1019c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final long[] duration = new long[1]; 1020c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1021c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project view.post(new Runnable() { 1022c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project public void run() { 1023c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project try { 1024c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project T[] data = operation.pre(); 1025c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project long start = Debug.threadCpuTimeNanos(); 1026c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project operation.run(data); 1027c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project duration[0] = Debug.threadCpuTimeNanos() - start; 1028c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project operation.post(data); 1029c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } finally { 1030c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project latch.countDown(); 1031c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1032c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1033c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project }); 1034c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1035c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project try { 1036c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project latch.await(CAPTURE_TIMEOUT, TimeUnit.MILLISECONDS); 1037c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } catch (InterruptedException e) { 1038c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project Log.w("View", "Could not complete the profiling of the view " + view); 1039c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project Thread.currentThread().interrupt(); 1040c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project return -1; 1041c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1042c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1043c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project return duration[0]; 1044c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1045c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1046223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy private static void captureLayers(View root, final DataOutputStream clientStream) 1047223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy throws IOException { 1048223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1049223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy try { 1050223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy Rect outRect = new Rect(); 1051223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy try { 1052223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy root.mAttachInfo.mSession.getDisplayFrame(root.mAttachInfo.mWindow, outRect); 1053223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } catch (RemoteException e) { 1054223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy // Ignore 1055223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1056223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1057223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.writeInt(outRect.width()); 1058223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.writeInt(outRect.height()); 1059223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 106065554f27855ce1764123604b061b10346f8b8404Romain Guy captureViewLayer(root, clientStream, true); 1061223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1062223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.write(2); 1063223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } finally { 1064223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.close(); 1065223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1066223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1067223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 106865554f27855ce1764123604b061b10346f8b8404Romain Guy private static void captureViewLayer(View view, DataOutputStream clientStream, boolean visible) 1069223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy throws IOException { 1070223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 107165554f27855ce1764123604b061b10346f8b8404Romain Guy final boolean localVisible = view.getVisibility() == View.VISIBLE && visible; 107265554f27855ce1764123604b061b10346f8b8404Romain Guy 1073223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy if ((view.mPrivateFlags & View.SKIP_DRAW) != View.SKIP_DRAW) { 1074223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy final int id = view.getId(); 1075223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy String name = view.getClass().getSimpleName(); 1076223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy if (id != View.NO_ID) { 1077223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy name = resolveId(view.getContext(), id).toString(); 1078223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1079223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1080223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.write(1); 1081223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.writeUTF(name); 108265554f27855ce1764123604b061b10346f8b8404Romain Guy clientStream.writeByte(localVisible ? 1 : 0); 1083223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1084223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy int[] position = new int[2]; 1085223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy // XXX: Should happen on the UI thread 1086223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy view.getLocationInWindow(position); 1087223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1088223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.writeInt(position[0]); 1089223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.writeInt(position[1]); 1090223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.flush(); 1091223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1092223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy Bitmap b = performViewCapture(view, true); 1093223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy if (b != null) { 1094223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy ByteArrayOutputStream arrayOut = new ByteArrayOutputStream(b.getWidth() * 1095223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy b.getHeight() * 2); 1096223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy b.compress(Bitmap.CompressFormat.PNG, 100, arrayOut); 1097223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.writeInt(arrayOut.size()); 1098223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy arrayOut.writeTo(clientStream); 1099223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1100223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.flush(); 1101223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1102223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1103223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy if (view instanceof ViewGroup) { 1104223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy ViewGroup group = (ViewGroup) view; 1105223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy int count = group.getChildCount(); 1106223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1107223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy for (int i = 0; i < count; i++) { 110865554f27855ce1764123604b061b10346f8b8404Romain Guy captureViewLayer(group.getChildAt(i), clientStream, localVisible); 1109223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1110223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1111223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1112223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1113c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project private static void capture(View root, final OutputStream clientStream, String parameter) 1114c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project throws IOException { 1115c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1116c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final View captureView = findView(root, parameter); 1117223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy Bitmap b = performViewCapture(captureView, false); 1118223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1119223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy if (b != null) { 1120223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy BufferedOutputStream out = null; 1121223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy try { 1122223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy out = new BufferedOutputStream(clientStream, 32 * 1024); 1123223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy b.compress(Bitmap.CompressFormat.PNG, 100, out); 1124223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy out.flush(); 1125223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } finally { 1126223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy if (out != null) { 1127223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy out.close(); 1128223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1129223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy b.recycle(); 1130223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1131223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } else { 1132223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy Log.w("View", "Failed to create capture bitmap!"); 1133223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy clientStream.close(); 1134223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 1135223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy } 11369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1137223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy private static Bitmap performViewCapture(final View captureView, final boolean skpiChildren) { 11389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (captureView != null) { 1139c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final CountDownLatch latch = new CountDownLatch(1); 11409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Bitmap[] cache = new Bitmap[1]; 11419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1142223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy captureView.post(new Runnable() { 11439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project public void run() { 11449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 1145958b9adc086f126dcd757d29f0d7f443ae9064b2Dianne Hackborn cache[0] = captureView.createSnapshot( 1146223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy Bitmap.Config.ARGB_8888, 0, skpiChildren); 1147958b9adc086f126dcd757d29f0d7f443ae9064b2Dianne Hackborn } catch (OutOfMemoryError e) { 1148958b9adc086f126dcd757d29f0d7f443ae9064b2Dianne Hackborn try { 1149958b9adc086f126dcd757d29f0d7f443ae9064b2Dianne Hackborn cache[0] = captureView.createSnapshot( 1150223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy Bitmap.Config.ARGB_4444, 0, skpiChildren); 1151958b9adc086f126dcd757d29f0d7f443ae9064b2Dianne Hackborn } catch (OutOfMemoryError e2) { 1152958b9adc086f126dcd757d29f0d7f443ae9064b2Dianne Hackborn Log.w("View", "Out of memory for bitmap"); 11539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } finally { 11559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project latch.countDown(); 11569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project }); 11599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 11619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project latch.await(CAPTURE_TIMEOUT, TimeUnit.MILLISECONDS); 1162223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy return cache[0]; 11639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (InterruptedException e) { 1164c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project Log.w("View", "Could not complete the capture of the view " + captureView); 1165c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project Thread.currentThread().interrupt(); 11669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1168223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy 1169223ff5c0586adbbd1d6d57a3a4d176222e8b7434Romain Guy return null; 11709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static void dump(View root, OutputStream clientStream) throws IOException { 11739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project BufferedWriter out = null; 11749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 117538e951b62ed956501b11f0715e23e3bf70942f6dRomain Guy out = new BufferedWriter(new OutputStreamWriter(clientStream, "utf-8"), 32 * 1024); 11769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project View view = root.getRootView(); 11779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (view instanceof ViewGroup) { 11789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project ViewGroup group = (ViewGroup) view; 1179105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project dumpViewHierarchyWithProperties(group.getContext(), group, out, 0); 11809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write("DONE."); 11829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.newLine(); 1183c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } catch (Exception e) { 1184c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project android.util.Log.w("View", "Problem dumping the view:", e); 11859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } finally { 11869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (out != null) { 11879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.close(); 11889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static View findView(ViewGroup group, String className, int hashCode) { 11939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (isRequestedView(group, className, hashCode)) { 11949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return group; 11959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 11969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 11979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final int count = group.getChildCount(); 11989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 11999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final View view = group.getChildAt(i); 12009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (view instanceof ViewGroup) { 12019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final View found = findView((ViewGroup) view, className, hashCode); 12029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (found != null) { 12039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return found; 12049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (isRequestedView(view, className, hashCode)) { 12069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return view; 12079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return null; 12119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static boolean isRequestedView(View view, String className, int hashCode) { 12149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return view.getClass().getName().equals(className) && view.hashCode() == hashCode; 12159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1217105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project private static void dumpViewHierarchyWithProperties(Context context, ViewGroup group, 12189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project BufferedWriter out, int level) { 1219105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project if (!dumpViewWithProperties(context, group, out, level)) { 12209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 12219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final int count = group.getChildCount(); 12249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 12259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final View view = group.getChildAt(i); 12269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (view instanceof ViewGroup) { 1227105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project dumpViewHierarchyWithProperties(context, (ViewGroup) view, out, level + 1); 12289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 1229105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project dumpViewWithProperties(context, view, out, level + 1); 12309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1234105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project private static boolean dumpViewWithProperties(Context context, View view, 1235105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project BufferedWriter out, int level) { 1236105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project 12379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 12389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < level; i++) { 12399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write(' '); 12409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write(view.getClass().getName()); 12429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write('@'); 12439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write(Integer.toHexString(view.hashCode())); 12449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write(' '); 1245105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project dumpViewProperties(context, view, out); 12469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.newLine(); 12479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 12489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.w("View", "Error while dumping hierarchy tree"); 12499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 12509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 12529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static Field[] getExportedPropertyFields(Class<?> klass) { 12559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sFieldsForClasses == null) { 12569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sFieldsForClasses = new HashMap<Class<?>, Field[]>(); 12579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1258c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (sAnnotations == null) { 1259c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project sAnnotations = new HashMap<AccessibleObject, ExportedProperty>(512); 1260c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1261c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 12629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final HashMap<Class<?>, Field[]> map = sFieldsForClasses; 1263c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final HashMap<AccessibleObject, ExportedProperty> annotations = sAnnotations; 12649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Field[] fields = map.get(klass); 12669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fields != null) { 12679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return fields; 12689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final ArrayList<Field> foundFields = new ArrayList<Field>(); 12719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fields = klass.getDeclaredFields(); 12729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = fields.length; 12749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 12759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Field field = fields[i]; 12769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (field.isAnnotationPresent(ExportedProperty.class)) { 12779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project field.setAccessible(true); 12789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project foundFields.add(field); 1279c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project annotations.put(field, field.getAnnotation(ExportedProperty.class)); 12809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fields = foundFields.toArray(new Field[foundFields.size()]); 12849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project map.put(klass, fields); 12859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return fields; 12879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 12889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 12899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static Method[] getExportedPropertyMethods(Class<?> klass) { 12909066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (sMethodsForClasses == null) { 1291c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project sMethodsForClasses = new HashMap<Class<?>, Method[]>(100); 1292c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1293c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (sAnnotations == null) { 1294c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project sAnnotations = new HashMap<AccessibleObject, ExportedProperty>(512); 12959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1296c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 12979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final HashMap<Class<?>, Method[]> map = sMethodsForClasses; 1298c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final HashMap<AccessibleObject, ExportedProperty> annotations = sAnnotations; 12999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Method[] methods = map.get(klass); 13019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (methods != null) { 13029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return methods; 13039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final ArrayList<Method> foundMethods = new ArrayList<Method>(); 13069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project methods = klass.getDeclaredMethods(); 1307a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 13089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = methods.length; 13099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 1310a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy final Method method = methods[i]; 13119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (method.getParameterTypes().length == 0 && 1312c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project method.isAnnotationPresent(ExportedProperty.class) && 1313c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project method.getReturnType() != Void.class) { 13149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project method.setAccessible(true); 13159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project foundMethods.add(method); 1316c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project annotations.put(method, method.getAnnotation(ExportedProperty.class)); 13179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project methods = foundMethods.toArray(new Method[foundMethods.size()]); 13219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project map.put(klass, methods); 13229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return methods; 13249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1326105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project private static void dumpViewProperties(Context context, Object view, 1327105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project BufferedWriter out) throws IOException { 1328105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project 1329105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project dumpViewProperties(context, view, out, ""); 13309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1332105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project private static void dumpViewProperties(Context context, Object view, 1333105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project BufferedWriter out, String prefix) throws IOException { 1334105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project 13359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Class<?> klass = view.getClass(); 13369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project do { 1338105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project exportFields(context, view, out, klass, prefix); 1339105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project exportMethods(context, view, out, klass, prefix); 13409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project klass = klass.getSuperclass(); 13419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } while (klass != Object.class); 13429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1343a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 1344105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project private static void exportMethods(Context context, Object view, BufferedWriter out, 1345105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project Class<?> klass, String prefix) throws IOException { 13469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Method[] methods = getExportedPropertyMethods(klass); 13489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = methods.length; 13509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 13519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Method method = methods[i]; 13529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //noinspection EmptyCatchBlock 13539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 13549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project // TODO: This should happen on the UI thread 13559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object methodValue = method.invoke(view, (Object[]) null); 13569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Class<?> returnType = method.getReturnType(); 13579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (returnType == int.class) { 1359c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final ExportedProperty property = sAnnotations.get(method); 1360105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project if (property.resolveId() && context != null) { 13619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final int id = (Integer) methodValue; 1362105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project methodValue = resolveId(context, id); 13639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 1364809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final FlagToString[] flagsMapping = property.flagMapping(); 1365809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy if (flagsMapping.length > 0) { 1366809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final int intValue = (Integer) methodValue; 1367809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final String valuePrefix = prefix + method.getName() + '_'; 1368809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy exportUnrolledFlags(out, flagsMapping, intValue, valuePrefix); 1369809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy } 1370809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 13719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final IntToString[] mapping = property.mapping(); 13729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mapping.length > 0) { 13739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final int intValue = (Integer) methodValue; 13749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project boolean mapped = false; 13759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int mappingCount = mapping.length; 13769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int j = 0; j < mappingCount; j++) { 13779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final IntToString mapper = mapping[j]; 13789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mapper.from() == intValue) { 13799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project methodValue = mapper.to(); 13809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mapped = true; 13819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 13829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 13859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!mapped) { 13869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project methodValue = intValue; 13879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 13899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1390c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } else if (returnType == int[].class) { 1391c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final ExportedProperty property = sAnnotations.get(method); 1392c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final int[] array = (int[]) methodValue; 1393c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final String valuePrefix = prefix + method.getName() + '_'; 1394c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final String suffix = "()"; 1395c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1396105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project exportUnrolledArray(context, out, property, array, valuePrefix, suffix); 13979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (!returnType.isPrimitive()) { 1398c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final ExportedProperty property = sAnnotations.get(method); 13999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (property.deepExport()) { 1400105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project dumpViewProperties(context, methodValue, out, prefix + property.prefix()); 14019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 14029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1405c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project writeEntry(out, prefix, method.getName(), "()", methodValue); 14069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IllegalAccessException e) { 14079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (InvocationTargetException e) { 14089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1412105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project private static void exportFields(Context context, Object view, BufferedWriter out, 1413105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project Class<?> klass, String prefix) throws IOException { 1414105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project 14159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Field[] fields = getExportedPropertyFields(klass); 14169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 14179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = fields.length; 14189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 14199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Field field = fields[i]; 14209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 14219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //noinspection EmptyCatchBlock 14229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 14239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object fieldValue = null; 14249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Class<?> type = field.getType(); 14259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 14269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (type == int.class) { 1427c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final ExportedProperty property = sAnnotations.get(field); 1428105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project if (property.resolveId() && context != null) { 14299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final int id = field.getInt(view); 1430105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project fieldValue = resolveId(context, id); 14319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 1432809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final FlagToString[] flagsMapping = property.flagMapping(); 1433809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy if (flagsMapping.length > 0) { 1434809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final int intValue = field.getInt(view); 1435809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final String valuePrefix = prefix + field.getName() + '_'; 1436809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy exportUnrolledFlags(out, flagsMapping, intValue, valuePrefix); 1437809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy } 1438809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 14399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final IntToString[] mapping = property.mapping(); 14409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mapping.length > 0) { 14419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final int intValue = field.getInt(view); 14429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int mappingCount = mapping.length; 14439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int j = 0; j < mappingCount; j++) { 14449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final IntToString mapped = mapping[j]; 14459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mapped.from() == intValue) { 14469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fieldValue = mapped.to(); 14479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project break; 14489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 14519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fieldValue == null) { 14529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fieldValue = intValue; 14539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1456c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } else if (type == int[].class) { 1457c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final ExportedProperty property = sAnnotations.get(field); 1458c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final int[] array = (int[]) field.get(view); 1459c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final String valuePrefix = prefix + field.getName() + '_'; 1460c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final String suffix = ""; 1461c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1462105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project exportUnrolledArray(context, out, property, array, valuePrefix, suffix); 1463c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1464c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project // We exit here! 1465c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project return; 14669066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else if (!type.isPrimitive()) { 1467c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final ExportedProperty property = sAnnotations.get(field); 14689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (property.deepExport()) { 1469105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project dumpViewProperties(context, field.get(view), out, 1470105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project prefix + property.prefix()); 14719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project continue; 14729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 14759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fieldValue == null) { 14769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fieldValue = field.get(view); 14779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1479c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project writeEntry(out, prefix, field.getName(), "", fieldValue); 14809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IllegalAccessException e) { 14819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14839066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 14849066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 1485c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project private static void writeEntry(BufferedWriter out, String prefix, String name, 1486c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project String suffix, Object value) throws IOException { 1487c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1488c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write(prefix); 1489c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write(name); 1490c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write(suffix); 1491c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write("="); 1492c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project writeValue(out, value); 1493c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write(' '); 1494c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1495c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1496809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy private static void exportUnrolledFlags(BufferedWriter out, FlagToString[] mapping, 1497809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy int intValue, String prefix) throws IOException { 1498809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 1499809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final int count = mapping.length; 1500809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy for (int j = 0; j < count; j++) { 1501809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final FlagToString flagMapping = mapping[j]; 1502809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final boolean ifTrue = flagMapping.outputIf(); 15035bcdff45bf4ada77ae7c95f520b795876adef75cRomain Guy final int maskResult = intValue & flagMapping.mask(); 15045bcdff45bf4ada77ae7c95f520b795876adef75cRomain Guy final boolean test = maskResult == flagMapping.equals(); 1505809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy if ((test && ifTrue) || (!test && !ifTrue)) { 1506809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy final String name = flagMapping.name(); 15075bcdff45bf4ada77ae7c95f520b795876adef75cRomain Guy final String value = "0x" + Integer.toHexString(maskResult); 1508809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy writeEntry(out, prefix, name, "", value); 1509809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy } 1510809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy } 1511809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy } 1512809a7f6080312f3e12f1a3a30eacf0e0c7627305Romain Guy 1513105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project private static void exportUnrolledArray(Context context, BufferedWriter out, 1514c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project ExportedProperty property, int[] array, String prefix, String suffix) 1515c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project throws IOException { 1516c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1517c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final IntToString[] indexMapping = property.indexMapping(); 1518c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final boolean hasIndexMapping = indexMapping.length > 0; 1519c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1520c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final IntToString[] mapping = property.mapping(); 1521c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final boolean hasMapping = mapping.length > 0; 1522c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1523105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project final boolean resolveId = property.resolveId() && context != null; 1524c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final int valuesCount = array.length; 1525c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1526c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project for (int j = 0; j < valuesCount; j++) { 1527c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project String name; 1528a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy String value = null; 1529c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1530c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final int intValue = array[j]; 1531c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1532c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project name = String.valueOf(j); 1533c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (hasIndexMapping) { 1534c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project int mappingCount = indexMapping.length; 1535c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project for (int k = 0; k < mappingCount; k++) { 1536c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final IntToString mapped = indexMapping[k]; 1537c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (mapped.from() == j) { 1538c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project name = mapped.to(); 1539c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project break; 1540c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1541c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1542c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1543c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1544c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (hasMapping) { 1545c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project int mappingCount = mapping.length; 1546c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project for (int k = 0; k < mappingCount; k++) { 1547c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project final IntToString mapped = mapping[k]; 1548c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (mapped.from() == intValue) { 1549c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project value = mapped.to(); 1550c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project break; 1551c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1552c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1553c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1554c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1555c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (resolveId) { 1556a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy if (value == null) value = (String) resolveId(context, intValue); 1557a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy } else { 1558a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy value = String.valueOf(intValue); 1559c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1560c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1561c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project writeEntry(out, prefix, name, suffix, value); 1562c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1563c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1564c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1565237c1ceea36024cf4194212e713806e3ce8a1c49Romain Guy static Object resolveId(Context context, int id) { 1566c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project Object fieldValue; 1567105925376f8d0f6b318c9938c7b83ef7fef094daThe Android Open Source Project final Resources resources = context.getResources(); 1568c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (id >= 0) { 1569c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project try { 1570c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project fieldValue = resources.getResourceTypeName(id) + '/' + 1571c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project resources.getResourceEntryName(id); 1572c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } catch (Resources.NotFoundException e) { 1573c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project fieldValue = "id/0x" + Integer.toHexString(id); 1574c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1575c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } else { 1576c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project fieldValue = "NO_ID"; 1577c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1578c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project return fieldValue; 1579c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1580c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 1581c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project private static void writeValue(BufferedWriter out, Object value) throws IOException { 1582c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project if (value != null) { 1583c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project String output = value.toString().replace("\n", "\\n"); 1584c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write(String.valueOf(output.length())); 1585c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write(","); 1586c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write(output); 1587c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } else { 1588c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project out.write("4,null"); 1589c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1590c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project } 1591c39a6e0c51e182338deb8b63d07933b585134929The Android Open Source Project 15929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static void dumpViewHierarchy(ViewGroup group, BufferedWriter out, int level) { 15939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (!dumpView(group, out, level)) { 15949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return; 15959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 15969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 15979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final int count = group.getChildCount(); 15989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 15999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final View view = group.getChildAt(i); 16009066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (view instanceof ViewGroup) { 16019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project dumpViewHierarchy((ViewGroup) view, out, level + 1); 16029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 16039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project dumpView(view, out, level + 1); 16049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16059066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static boolean dumpView(Object view, BufferedWriter out, int level) { 16099066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 16109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < level; i++) { 16119066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write(' '); 16129066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write(view.getClass().getName()); 16149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write('@'); 16159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.write(Integer.toHexString(view.hashCode())); 16169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project out.newLine(); 16179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IOException e) { 16189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Log.w("View", "Error while dumping hierarchy tree"); 16199066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return false; 16209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return true; 16229066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16249066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static Field[] capturedViewGetPropertyFields(Class<?> klass) { 16259066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mCapturedViewFieldsForClasses == null) { 16269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mCapturedViewFieldsForClasses = new HashMap<Class<?>, Field[]>(); 16279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final HashMap<Class<?>, Field[]> map = mCapturedViewFieldsForClasses; 16299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16309066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Field[] fields = map.get(klass); 16319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fields != null) { 16329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return fields; 16339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16349066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final ArrayList<Field> foundFields = new ArrayList<Field>(); 16369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fields = klass.getFields(); 16379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = fields.length; 16399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 16409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Field field = fields[i]; 16419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (field.isAnnotationPresent(CapturedViewProperty.class)) { 16429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project field.setAccessible(true); 16439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project foundFields.add(field); 16449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project fields = foundFields.toArray(new Field[foundFields.size()]); 16489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project map.put(klass, fields); 16499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return fields; 16519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static Method[] capturedViewGetPropertyMethods(Class<?> klass) { 16549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (mCapturedViewMethodsForClasses == null) { 16559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project mCapturedViewMethodsForClasses = new HashMap<Class<?>, Method[]>(); 16569066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final HashMap<Class<?>, Method[]> map = mCapturedViewMethodsForClasses; 16589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Method[] methods = map.get(klass); 16609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (methods != null) { 16619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return methods; 16629066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16649066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final ArrayList<Method> foundMethods = new ArrayList<Method>(); 16659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project methods = klass.getMethods(); 1666a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 16679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = methods.length; 16689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 1669a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy final Method method = methods[i]; 16709066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (method.getParameterTypes().length == 0 && 16719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project method.isAnnotationPresent(CapturedViewProperty.class) && 16729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project method.getReturnType() != Void.class) { 16739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project method.setAccessible(true); 16749066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project foundMethods.add(method); 16759066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 16779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16789066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project methods = foundMethods.toArray(new Method[foundMethods.size()]); 16799066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project map.put(klass, methods); 16809066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16819066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return methods; 16829066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1683a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 1684a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy private static String capturedViewExportMethods(Object obj, Class<?> klass, 16859066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project String prefix) { 16869066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16879066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (obj == null) { 16889066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return "null"; 16899066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1690a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 16919066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project StringBuilder sb = new StringBuilder(); 16929066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Method[] methods = capturedViewGetPropertyMethods(klass); 16939066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 16949066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = methods.length; 16959066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 16969066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Method method = methods[i]; 16979066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 16989066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object methodValue = method.invoke(obj, (Object[]) null); 16999066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Class<?> returnType = method.getReturnType(); 1700a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 17019066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project CapturedViewProperty property = method.getAnnotation(CapturedViewProperty.class); 17029066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (property.retrieveReturn()) { 17039066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //we are interested in the second level data only 17049066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append(capturedViewExportMethods(methodValue, returnType, method.getName() + "#")); 1705a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy } else { 17069066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append(prefix); 17079066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append(method.getName()); 17089066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append("()="); 1709a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 17109066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (methodValue != null) { 1711a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy final String value = methodValue.toString().replace("\n", "\\n"); 1712a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy sb.append(value); 17139066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 17149066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append("null"); 17159066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 17169066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append("; "); 17179066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 17189066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IllegalAccessException e) { 1719a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy //Exception IllegalAccess, it is OK here 17209066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //we simply ignore this method 17219066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (InvocationTargetException e) { 1722a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy //Exception InvocationTarget, it is OK here 17239066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //we simply ignore this method 1724a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy } 1725a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy } 17269066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return sb.toString(); 17279066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 17289066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 17299066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project private static String capturedViewExportFields(Object obj, Class<?> klass, String prefix) { 1730a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 17319066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (obj == null) { 17329066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return "null"; 17339066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1734a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 17359066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project StringBuilder sb = new StringBuilder(); 17369066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Field[] fields = capturedViewGetPropertyFields(klass); 17379066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 17389066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project int count = fields.length; 17399066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project for (int i = 0; i < count; i++) { 17409066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final Field field = fields[i]; 17419066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project try { 17429066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Object fieldValue = field.get(obj); 17439066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 17449066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append(prefix); 17459066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append(field.getName()); 17469066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append("="); 17479066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project 17489066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project if (fieldValue != null) { 17499066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project final String value = fieldValue.toString().replace("\n", "\\n"); 17509066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append(value); 17519066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } else { 17529066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append("null"); 17539066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 17549066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append(' '); 17559066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } catch (IllegalAccessException e) { 1756a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy //Exception IllegalAccess, it is OK here 17579066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project //we simply ignore this field 17589066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 17599066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 17609066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project return sb.toString(); 17619066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 1762a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy 17639066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project /** 1764a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy * Dump view info for id based instrument test generation 17659066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * (and possibly further data analysis). The results are dumped 1766a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy * to the log. 17679066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param tag for log 17689066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project * @param view for dump 17699066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project */ 1770a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy public static void dumpCapturedView(String tag, Object view) { 17719066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project Class<?> klass = view.getClass(); 17729066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project StringBuilder sb = new StringBuilder(klass.getName() + ": "); 17739066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project sb.append(capturedViewExportFields(view, klass, "")); 1774a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy sb.append(capturedViewExportMethods(view, klass, "")); 1775a1f3e4aef19882b4b81075d9205bd363efe1e66dRomain Guy Log.d(tag, sb.toString()); 17769066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project } 17779066cfe9886ac131c34d59ed0e2d287b0e3c0087The Android Open Source Project} 1778