com.android.utils.TreeDebug.java Source code

Java tutorial

Introduction

Here is the source code for com.android.utils.TreeDebug.java

Source

/*
 * 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.android.utils;

import android.graphics.Rect;
import android.support.v4.view.accessibility.AccessibilityNodeInfoCompat;
import android.text.TextUtils;
import android.util.Log;

import java.util.HashSet;

/**
 * Util class to help debug Node trees.
 */
public class TreeDebug {

    /**
     * Logs the tree using the input node as the root.
     */
    public static void logNodeTree(AccessibilityNodeInfoCompat node) {
        if (node == null) {
            return;
        }

        HashSet<AccessibilityNodeInfoCompat> seen = new HashSet<>();
        logNodeTree(AccessibilityNodeInfoCompat.obtain(node), "", seen);
        for (AccessibilityNodeInfoCompat n : seen) {
            n.recycle();
        }
    }

    private static void logNodeTree(AccessibilityNodeInfoCompat node, String indent,
            HashSet<AccessibilityNodeInfoCompat> seen) {
        if (!seen.add(node)) {
            LogUtils.log(TreeDebug.class, Log.VERBOSE, "Cycle: %d", node.hashCode());
            return;
        }

        // Include the hash code as a "poor man's" id, knowing that it
        // might not always be unique.
        LogUtils.log(TreeDebug.class, Log.VERBOSE, "%s(%d)%s", indent, node.hashCode(), nodeDebugDescription(node));

        indent += "  ";
        int childCount = node.getChildCount();
        for (int i = 0; i < childCount; ++i) {
            AccessibilityNodeInfoCompat child = node.getChild(i);
            if (child == null) {
                LogUtils.log(TreeDebug.class, Log.VERBOSE, "%sCouldn't get child %d", indent, i);
                continue;
            }

            logNodeTree(child, indent, seen);
        }
    }

    private static void appendSimpleName(StringBuilder sb, CharSequence fullName) {
        int dotIndex = TextUtils.lastIndexOf(fullName, '.');
        if (dotIndex < 0) {
            dotIndex = 0;
        }

        sb.append(fullName, dotIndex, fullName.length());
    }

    /**
     * Gets a description of the properties of a node.
     */
    private static CharSequence nodeDebugDescription(AccessibilityNodeInfoCompat node) {
        StringBuilder sb = new StringBuilder();
        sb.append(node.getWindowId());

        if (node.getClassName() != null) {
            appendSimpleName(sb, node.getClassName());
        } else {
            sb.append("??");
        }

        if (!node.isVisibleToUser()) {
            sb.append(":invisible");
        }

        Rect rect = new Rect();
        node.getBoundsInScreen(rect);
        sb.append(":");
        sb.append("(").append(rect.left).append(", ").append(rect.top).append(" - ").append(rect.right).append(", ")
                .append(rect.bottom).append(")");

        if (node.getText() != null) {
            sb.append(":");
            sb.append(node.getText().toString().trim());
        }

        if (node.getContentDescription() != null) {
            sb.append(":");
            sb.append(node.getContentDescription().toString().trim());
        }

        int actions = node.getActions();
        if (actions != 0) {
            sb.append(":");
            if ((actions & AccessibilityNodeInfoCompat.ACTION_FOCUS) != 0) {
                sb.append("F");
            }
            if ((actions & AccessibilityNodeInfoCompat.ACTION_ACCESSIBILITY_FOCUS) != 0) {
                sb.append("A");
            }
            if ((actions & AccessibilityNodeInfoCompat.ACTION_CLEAR_ACCESSIBILITY_FOCUS) != 0) {
                sb.append("a");
            }
            if ((actions & AccessibilityNodeInfoCompat.ACTION_SCROLL_BACKWARD) != 0) {
                sb.append("-");
            }
            if ((actions & AccessibilityNodeInfoCompat.ACTION_CLICK) != 0) {
                sb.append("C");
            }
            if ((actions & AccessibilityNodeInfoCompat.ACTION_LONG_CLICK) != 0) {
                sb.append("L");
            }
            if ((actions & AccessibilityNodeInfoCompat.ACTION_SCROLL_FORWARD) != 0) {
                sb.append("+");
            }
            if ((actions & AccessibilityNodeInfoCompat.ACTION_EXPAND) != 0) {
                sb.append("e");
            }
            if ((actions & AccessibilityNodeInfoCompat.ACTION_COLLAPSE) != 0) {
                sb.append("c");
            }
        }

        if (node.isCheckable()) {
            sb.append(":");
            if (node.isChecked()) {
                sb.append("(X)");
            } else {
                sb.append("( )");
            }
        }

        if (node.isFocusable()) {
            sb.append(":focusable");
        }

        if (node.isFocused()) {
            sb.append(":focused");
        }

        if (node.isSelected()) {
            sb.append(":selected");
        }

        if (node.isClickable()) {
            sb.append(":clickable");
        }

        if (node.isLongClickable()) {
            sb.append(":longClickable");
        }

        if (node.isAccessibilityFocused()) {
            sb.append(":accessibilityFocused");
        }

        if (!node.isEnabled()) {
            sb.append(":disabled");
        }

        if (node.getCollectionInfo() != null) {
            sb.append(":collection");
            sb.append("#R");
            sb.append(node.getCollectionInfo().getRowCount());
            sb.append("C");
            sb.append(node.getCollectionInfo().getColumnCount());
        }

        if (node.getCollectionItemInfo() != null) {
            if (node.getCollectionItemInfo().isHeading()) {
                sb.append(":heading");
            } else {
                sb.append(":item");
            }

            sb.append("#r");
            sb.append(node.getCollectionItemInfo().getRowIndex());
            sb.append("c");
            sb.append(node.getCollectionItemInfo().getColumnIndex());
        }

        return sb.toString();
    }
}