得到Android无埋点方案细节分析

开源代码:DDAutoTracker

1. View唯一标识

id组成:ActivityName_LayoutFileName_idName

对应源码:ResourceHelper#getGlobalIdName

public static String getGlobalIdName(@NonNull View view) {
        int id = view.getId();
        ...
        try {
            Context context = view.getContext();
            // 获取activityName
            String activityName = context.getClass().getSimpleName();
            // 获取布局文件名
            String layoutFileName = getLayoutFileName(view);
            String idName;
            ...
            // 获取id资源名
            idName = getResourceEntryName(context, id);
            ...

            return String.format("%s_%s_%s", activityName, layoutFileName, idName);
            ...
        }

    }

2. 定位交互控件:TouchTarget方案

TouchTarget如何赋值?

一次简单的单点触控交互流程是这样的:

ACTION_DOWN(手指落下)
ACTION_MOVE(移动)
ACTION_MOVE
ACTION_MOVE
ACTION_MOVE
...
ACTION_UP(离开)

当用户触发ACTION_DOWN事件时,会执行如下逻辑,寻找消费当前事件的TouchTarget。

if (actionMasked == MotionEvent.ACTION_DOWN){
    //如果是down事件,遍历child,找到TouchTarget
    ..
    ..
    final View[] children = mChildren;
    for (int i = childrenCount - 1; i >= 0; i--) {
       final int childIndex = getAndVerifyPreorderedIndexchildrenCount, i, customOrder);
       final View child = getAndVerifyPreorderedView(preorderedList, children, childIndex);
       ..
       ..
       if (dispatchTransformedTouchEvent(ev, false, child, idBitsToAssign)) {
          // child 消费了触摸事件
          ..
          ..
          // 根据消费了触摸事件的View创建TouchTarget
           newTouchTarget = addTouchTarget(child, idBitsToAssign);
          ..
          ..
          break;
      }
}

假设ACTION_DOWN事件分发路径如下:
ViewGroup1
ViewGroup2
ViewGroup3
View

dispatch-path.png

路径中的每个ViewGroup都维护一个mFirstTouchTarget。

    // First touch target in the linked list of touch targets.
    private TouchTarget mFirstTouchTarget;

ACTION_DOWN事件分发过程中,路径中各ViewGroup成员变量mFirstTouchTarget赋值流程如下:

mFirstTouchTarget赋值流程

何时读取目标控件?

【得到Android】通过在Activity的window上调用window.setCallback() 接管窗口的事件派发,并在dispatchTouchEvent处理函数中添加读取目标控件的处理逻辑。如果接收到up事件,执行处理逻辑,通过ViewGroup TouchTarget链表,找到本次交互行为的目标控件。

读取目标控件的处理逻辑核心代码如下:

    private View findActionTargets() {
        ViewGroup decorView = (ViewGroup) getWindow().getDecorView();
        int content_id = android.R.id.content;
        ViewGroup content = (ViewGroup) decorView.findViewById(content_id);
        if (content == null) {
            content = decorView; //对于非Activity DecorView 的情况处理
        }
        View touchTarget;

        ViewGroup vg = content;
        while (true) {
            //获取指定vg的mFirstTouchTarget.child
            touchTarget = ViewHelper.findTouchTarget(vg);

            //无法找到touchTarget 相关信息
            if (touchTarget == null) return null;

            //已经找到touchTarget
            if (touchTarget == vg) break;

            boolean isVG = touchTarget instanceof ViewGroup;
            //已经找到touchTarget
            if (!isVG) break;

            //未找到touchTarget
            vg = (ViewGroup) touchTarget;
        }

        return touchTarget;
    }

其中,ViewHelper.findTouchTarget(ViewGroup)通过反射获取指定ViewGroup的mFirstTouchTarget.child,源码如下:

public static View findTouchTarget(@NonNull ViewGroup ancestor) {
        Preconditions.checkNotNull(ancestor);

        try {
            Field firstTouchTargetField = CoreUtils.getDeclaredField(ancestor, "mFirstTouchTarget");
            if (firstTouchTargetField == null) {
                logReflectException("mFirstTouchTarget");
                return ancestor;
            }

            firstTouchTargetField.setAccessible(true);
            Object firstTouchTarget = firstTouchTargetField.get(ancestor);
            if (firstTouchTarget == null) return ancestor;

            Field firstTouchViewField = firstTouchTarget.getClass().getDeclaredField("child");
            if (firstTouchViewField == null) {
                logReflectException("child");
                return ancestor;
            }

            firstTouchViewField.setAccessible(true);
            View firstTouchView = (View) firstTouchViewField.get(firstTouchTarget);
            if (firstTouchView == null) return ancestor;

            return firstTouchView;

        } catch (Exception e) {
            e.printStackTrace();

            return null;
        }
    }

再来看一下ViewGroup内mFirstTouchTarget赋值流程图,结合上面的算法与赋值流程图,可以反过来获取目标控件。


mFirstTouchTarget赋值流程

下述伪代码解释了目标控件获取过程:
ViewGroup2 = ViewHelper.findTouchTarget(ViewGroup1);
ViewGroup3 = ViewHelper.findTouchTarget(ViewGroup2);
View = ViewHelper.findTouchTarget(ViewGroup3);
View即为目标控件。

3. 布局文件名的获取

对应源码:ResourceHelper#getLayoutFileName

    private static String getLayoutFileName(@NonNull View view) {

        String idNameSpace = (String) view.getTag(R.id.id_namespace_tag);
        if (!TextUtils.isEmpty(idNameSpace)) return idNameSpace;

        View tmp = view;
        while (tmp.getParent() != null && (tmp.getParent() instanceof View)) {
            View parent = (View) tmp.getParent();

            String space = (String) parent.getTag(R.id.id_namespace_tag);
            if (!TextUtils.isEmpty(space)) return space;

            tmp = parent;
        }

        return "";
    }

算法思路:从view出发,向上回溯,读取ancestor的tag,读到则跳出循环。其中tag即为布局文件名。

Tag是如何塞进去的呢?

复习:

public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot);

  • 当root不为null,attachToRoot为true时,表示将resource指定的布局添加到root中,添加的过程中resource所指定的的布局的根节点的各个属性都是有效的;

  • 如果root不为null,而attachToRoot为false的话,表示不将第一个参数所指定的View添加到root中;

  • 当root为null时,不论attachToRoot为true还是为false,显示效果都是一样的。当root为null表示我不需要将第一个参数所指定的布局添加到任何容器中,同时也表示没有任何容器来来协助第一个参数所指定布局的根节点生成布局参数。

方案:包装系统的mInflater,在调用inflate时,设置tag。

对应源码:LayoutInflaterWrapper#inflate

public View inflate(@LayoutRes int resource, @Nullable ViewGroup root, boolean attachToRoot) {
        final Resources res = getContext().getResources();

        final XmlResourceParser parser = res.getLayout(resource);
        try {
            View view = inflate(parser, root, attachToRoot);

            attachToRoot = (attachToRoot && root != null);
            if (!attachToRoot) {
                view.setTag(R.id.id_namespace_tag, ResourceHelper.getResourceEntryName(getContext(), resource));
                return view;
            }

            int childCount = root.getChildCount();
            View tagedView = root.getChildAt(childCount - 1);
            tagedView.setTag(R.id.id_namespace_tag, ResourceHelper.getResourceEntryName(getContext(), resource));
            return view;
        } finally {
            parser.close();
        }
    }

算法思路:

  1. 调用系统inflate,返回一个view;

  2. attachToRoot为false,resource指定的布局对应的view即为返回的view,将resource指定的布局文件名称设置为该view的tag;

  3. attachToRoot为true,resource指定的布局对应的view为返回的view的最后一个child;将resource指定的布局文件名称设置为该child的tag。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容