com.tmall.wireless.tangram.support.SimpleClickSupport.java Source code

Java tutorial

Introduction

Here is the source code for com.tmall.wireless.tangram.support.SimpleClickSupport.java

Source

/*
 * MIT License
 *
 * Copyright (c) 2017 Alibaba Group
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */

package com.tmall.wireless.tangram.support;

import android.support.v4.util.ArrayMap;
import android.util.Log;
import android.view.View;

import com.tmall.wireless.tangram.dataparser.concrete.Cell;
import com.tmall.wireless.tangram.structure.BaseCell;
import com.tmall.wireless.tangram.util.LogUtils;

import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * ClickSupport which provide a group of methods to handle click events on ComponentViews
 * <p>
 * Usage:
 * <p>
 * <pre>
 *     SimpleClickSupport support = serviceManager.getService(SimpleClickSupport.class)
 *     support.onClick(targetView, thisCell, eventType, [params]);
 * </pre>
 * <p>
 * Which make all the click handlers registered in one place, and decouple the business logic from ComponentViews
 */
public abstract class SimpleClickSupport {
    private static final String TAG = "SimpleClickSupport";

    private static final String ON_CLICK_METHOD_NAME = "onClick";
    private static final String ON_CLICK_METHOD_PREFIX = "on";
    private static final String ON_CLICK_METHOD_POSTFIX = "Click";

    /**
     * Ignore bridge or synthetic methods that added by compiler
     */
    private static final int BRIDGE = 0x40;
    private static final int SYNTHETIC = 0x1000;

    private static final int MODIFIERS_IGNORE = Modifier.ABSTRACT | Modifier.STATIC | BRIDGE | SYNTHETIC;

    private final Map<Class<?>, OnClickMethod> mOnClickMethods = new ArrayMap<>();

    private boolean optimizedMode;

    public SimpleClickSupport() {
    }

    /**
     * enable opt mode, users route click handler by themselves, otherwise click handler is routed by inner and invoked by reflection
     * @param optimizedMode true to enable opt mode
     */
    public void setOptimizedMode(boolean optimizedMode) {
        this.optimizedMode = optimizedMode;
    }

    private void findClickMethods(Method[] methods) {
        for (Method method : methods) {
            String methodName = method.getName();
            if (!methodName.equals(ON_CLICK_METHOD_NAME) && methodName.startsWith(ON_CLICK_METHOD_NAME)
                    || (methodName.startsWith(ON_CLICK_METHOD_PREFIX)
                            && methodName.endsWith(ON_CLICK_METHOD_POSTFIX))) {
                int modifiers = method.getModifiers();
                if ((modifiers & Modifier.PUBLIC) != 0 && (modifiers & MODIFIERS_IGNORE) == 0) {
                    Class<?>[] parameterTypes = method.getParameterTypes();
                    if (parameterTypes.length == 3 || parameterTypes.length == 4) {
                        Class<?> viewType = parameterTypes[0];
                        Class<?> cellType = parameterTypes[1];
                        Class<?> clickIntType = parameterTypes[2];
                        if (View.class.isAssignableFrom(viewType) && BaseCell.class.isAssignableFrom(cellType)
                                && (clickIntType.equals(int.class) || clickIntType.equals(Integer.class))) {
                            if (parameterTypes.length == 4) {
                                Class<?> clickParamsType = parameterTypes[3];
                                if (Map.class.isAssignableFrom(clickParamsType)) {
                                    mOnClickMethods.put(viewType, new OnClickMethod(4, method));
                                }
                            } else {
                                mOnClickMethods.put(viewType, new OnClickMethod(3, method));
                            }
                        }
                    }
                }
            }
        }
    }

    /**
     * Handler click event on item
     *
     * @param targetView the view that trigger the click event, not the view respond the cell!
     * @param cell       the corresponding cell
     * @param type       click event type, defined by developer.
     */
    public void onClick(View targetView, BaseCell cell, int type) {
        if (cell instanceof Cell) {
            onClick(targetView, (Cell) cell, type);
        } else {
            onClick(targetView, cell, type, null);
        }
    }

    public void onClick(View targetView, Cell cell, int type) {
        onClick(targetView, cell, type, null);
    }

    public void onClick(View targetView, BaseCell cell, int type, Map<String, Object> params) {
        if (optimizedMode) {
            defaultClick(targetView, cell, type);
        } else {
            if (mOnClickMethods.isEmpty()) {
                findClickMethods(this.getClass().getMethods());
            }
            List<Class<?>> classes = lookupCellTypes(targetView.getClass());
            for (Class clz : classes) {
                if (clz.equals(View.class)) {
                    continue;
                }
                if (mOnClickMethods.containsKey(clz)) {
                    OnClickMethod onClickMethod = mOnClickMethods.get(clz);
                    try {
                        if (onClickMethod.paramLength == 3) {
                            onClickMethod.method.invoke(this, targetView, cell, type);
                            return;
                        } else if (onClickMethod.paramLength == 4) {
                            onClickMethod.method.invoke(this, targetView, cell, type, params);
                            return;
                        }
                    } catch (Exception e) {
                        LogUtils.e(TAG, "Invoke onClick method error: " + Log.getStackTraceString(e), e);
                    }
                }
            }
            defaultClick(targetView, cell, type);
        }
    }

    public void defaultClick(View targetView, BaseCell cell, int type) {

    }

    private static final Map<Class<?>, List<Class<?>>> cellTypesCache = new ConcurrentHashMap<>();

    private List<Class<?>> lookupCellTypes(Class<?> cellClass) {
        List<Class<?>> eventTypes = cellTypesCache.get(cellClass);
        if (eventTypes == null) {
            eventTypes = new ArrayList<>();
            Class<?> clazz = cellClass;
            while (clazz != null && !clazz.equals(BaseCell.class)) {
                eventTypes.add(clazz);
                clazz = clazz.getSuperclass();
            }
            cellTypesCache.put(cellClass, eventTypes);
        }
        return eventTypes;
    }

    static class OnClickMethod {
        int paramLength;
        Method method;

        public OnClickMethod(int paramLength, Method method) {
            this.paramLength = paramLength;
            this.method = method;
        }
    }

}