| /* |
| * Copyright (C) 2014 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| package com.google.android.apps.common.testing.ui.espresso.util; |
| |
| import static com.google.android.apps.common.testing.ui.espresso.util.TreeIterables.depthFirstViewTraversalWithDistance; |
| |
| import com.google.android.apps.common.testing.ui.espresso.util.TreeIterables.ViewAndDistance; |
| import com.google.common.base.Function; |
| import com.google.common.base.Joiner; |
| import com.google.common.base.Objects; |
| import com.google.common.base.Objects.ToStringHelper; |
| import com.google.common.base.Preconditions; |
| import com.google.common.base.Strings; |
| import com.google.common.collect.Iterables; |
| |
| import android.content.res.Resources; |
| import android.os.Build; |
| import android.util.Printer; |
| import android.util.StringBuilderPrinter; |
| import android.view.View; |
| import android.view.ViewGroup; |
| import android.view.inputmethod.EditorInfo; |
| import android.view.inputmethod.InputConnection; |
| import android.widget.Checkable; |
| import android.widget.TextView; |
| |
| import java.util.List; |
| |
| /** |
| * Text converters for various Android objects. |
| */ |
| public final class HumanReadables { |
| |
| private HumanReadables() {} |
| |
| /** |
| * Prints out an error message feature the view hierarchy starting at the rootView. |
| * |
| * @param rootView the root of the hierarchy tree to print out. |
| * @param problemViews list of the views that you would like to point out are causing the error |
| * message or null, if you want to skip this feature. |
| * @param errorHeader the header of the error message (should contain the description of why the |
| * error is happening). |
| * @param problemViewSuffix the message to append to the view description in the tree printout. |
| * Required if problemViews is supplied. Otherwise, null is acceptable. |
| * @return a string for human consumption. |
| */ |
| public static String getViewHierarchyErrorMessage(View rootView, |
| final List<View> problemViews, |
| String errorHeader, |
| final String problemViewSuffix) { |
| Preconditions.checkArgument(problemViews == null || problemViewSuffix != null); |
| StringBuilder errorMessage = new StringBuilder(errorHeader); |
| if (problemViewSuffix != null) { |
| errorMessage.append( |
| String.format("\nProblem views are marked with '%s' below.", problemViewSuffix)); |
| } |
| |
| errorMessage.append("\n\nView Hierarchy:\n"); |
| |
| Joiner.on("\n").appendTo(errorMessage, Iterables.transform( |
| depthFirstViewTraversalWithDistance(rootView), new Function<ViewAndDistance, String>() { |
| @Override |
| public String apply(ViewAndDistance viewAndDistance) { |
| String formatString = "+%s%s "; |
| if (problemViews != null |
| && problemViews.contains(viewAndDistance.getView())) { |
| formatString += problemViewSuffix; |
| } |
| formatString += "\n|"; |
| |
| return String.format(formatString, |
| Strings.padStart(">", viewAndDistance.getDistanceFromRoot() + 1, '-'), |
| HumanReadables.describe(viewAndDistance.getView())); |
| } |
| })); |
| |
| return errorMessage.toString(); |
| } |
| |
| /** |
| * Transforms an arbitrary view into a string with (hopefully) enough debug info. |
| * |
| * @param v nullable view |
| * @return a string for human consumption. |
| */ |
| public static String describe(View v) { |
| if (null == v) { |
| return "null"; |
| } |
| ToStringHelper helper = Objects.toStringHelper(v).add("id", v.getId()); |
| if (v.getId() != -1 && v.getResources() != null) { |
| try { |
| helper.add("res-name", v.getResources().getResourceEntryName(v.getId())); |
| } catch (Resources.NotFoundException ignore) { |
| // Do nothing. |
| } |
| } |
| if (null != v.getContentDescription()) { |
| helper.add("desc", v.getContentDescription()); |
| } |
| |
| switch (v.getVisibility()) { |
| case View.GONE: |
| helper.add("visibility", "GONE"); |
| break; |
| case View.INVISIBLE: |
| helper.add("visibility", "INVISIBLE"); |
| break; |
| case View.VISIBLE: |
| helper.add("visibility", "VISIBLE"); |
| break; |
| default: |
| helper.add("visibility", v.getVisibility()); |
| } |
| |
| helper.add("width", v.getWidth()) |
| .add("height", v.getHeight()) |
| .add("has-focus", v.hasFocus()) |
| .add("has-focusable", v.hasFocusable()) |
| .add("has-window-focus", v.hasWindowFocus()) |
| .add("is-clickable", v.isClickable()) |
| .add("is-enabled", v.isEnabled()) |
| .add("is-focused", v.isFocused()) |
| .add("is-focusable", v.isFocusable()) |
| .add("is-layout-requested", v.isLayoutRequested()) |
| .add("is-selected", v.isSelected()); |
| |
| if (null != v.getRootView()) { |
| // pretty much only true in unit-tests. |
| helper.add("root-is-layout-requested", v.getRootView().isLayoutRequested()); |
| } |
| |
| EditorInfo ei = new EditorInfo(); |
| InputConnection ic = v.onCreateInputConnection(ei); |
| boolean hasInputConnection = ic != null; |
| helper.add("has-input-connection", hasInputConnection); |
| if (hasInputConnection) { |
| StringBuilder sb = new StringBuilder(); |
| sb.append("["); |
| Printer p = new StringBuilderPrinter(sb); |
| ei.dump(p, ""); |
| sb.append("]"); |
| helper.add("editor-info", sb.toString().replace("\n", " ")); |
| } |
| |
| if (Build.VERSION.SDK_INT > 10) { |
| helper.add("x", v.getX()).add("y", v.getY()); |
| } |
| |
| if (v instanceof TextView) { |
| innerDescribe((TextView) v, helper); |
| } |
| if (v instanceof Checkable) { |
| innerDescribe((Checkable) v, helper); |
| } |
| if (v instanceof ViewGroup) { |
| innerDescribe((ViewGroup) v, helper); |
| } |
| return helper.toString(); |
| } |
| |
| private static void innerDescribe(TextView textBox, ToStringHelper helper) { |
| if (null != textBox.getText()) { |
| helper.add("text", textBox.getText()); |
| } |
| |
| if (null != textBox.getError()) { |
| helper.add("error-text", textBox.getError()); |
| } |
| |
| if (null != textBox.getHint()) { |
| helper.add("hint", textBox.getHint()); |
| } |
| |
| helper.add("input-type", textBox.getInputType()); |
| helper.add("ime-target", textBox.isInputMethodTarget()); |
| } |
| |
| private static void innerDescribe(Checkable checkable, ToStringHelper helper) { |
| helper.add("is-checked", checkable.isChecked()); |
| } |
| |
| private static void innerDescribe(ViewGroup viewGroup, ToStringHelper helper) { |
| helper.add("child-count", viewGroup.getChildCount()); |
| } |
| } |