android事件分发机制,如果大家不熟悉的话,很难去做好自定义控件的事件处理。最近在读《android自定义控件高级进阶与精彩实例》这本书,第4章消息处理写的挺好的。以下都是我摘抄自该书的笔记。
1. 概述
事件分发主要涉及3个函数:
public boolean dispatchTouchEvent(MotionEvent event)
public boolean onInterceptTouchEvent(MotionEvent event)
public boolean onTouchEvent(MotionEvent event)
event可以通过int action = event.getAction()
获取具体的事件,action取值如下:
// 手指按下时发出的消息
public static final int ACTION_DOWN = 0;
// 手指抬起时发出的消息
public static final int ACTION_UP = 1;
// 手指移动时发出的消息
public static final int ACTION_MOVE = 2;
// 结束事件时发出的消息,系统在一定条件下自动发出的
public static final int ACTION_CANCEL = 3;
为了研究这个事件流程,我们需要创建一个demo,通过log来查看一些事件的输出来验证事件的分发流程。
1.1 创建demo
创建2个activity,一个叫MainActivity, 一个叫SecondActivity:
MainActivity.java
package com.touch.harvic.testtouchevent;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnClickListener;
import androidx.appcompat.app.AppCompatActivity;
public class MainActivity extends AppCompatActivity implements OnClickListener{
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
findViewById(R.id.second_activity).setOnClickListener(this);
}
@Override
public void onClick(View v) {
int id = v.getId();
switch (id){
case R.id.second_activity:{
startActivity(new Intent(MainActivity.this,SecondActivity.class));
}
break;
}
}
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
LogHelper.onLog("MainActivity dispatchTouchEvent",event);
return super.dispatchTouchEvent(event);
}
@Override
public boolean onTouchEvent(MotionEvent event) {
LogHelper.onLog("MainActivity onTouchEvent",event);
return super.onTouchEvent(event);
}
}
布局文件activity_main.xml:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center"
tools:context=".MainActivity">
<Button
android:id="@+id/second_activity"
android:padding="10dp"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:text="Second Activity"/>
</LinearLayout>
SecondActivity
package com.touch.harvic.testtouchevent;
import android.os.Bundle;
import android.util.Log;
import android.view.MotionEvent;
import androidx.appcompat.app.AppCompatActivity;
public class SecondActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
}
@Override
public boolean dispatchTouchEvent(MotionEvent event) {
LogHelper.onLog("SecondActivity dispatchTouchEvent",event);
// return false;
return super.dispatchTouchEvent(event);
// return true;
}
@Override
public boolean onTouchEvent(MotionEvent event) {
LogHelper.onLog("SecondActivity onTouchEvent",event);
return super.onTouchEvent(event);
}
}
布局文件activity_second.xml:
<?xml version="1.0" encoding="utf-8"?>
<com.touch.harvic.testtouchevent.CustomFirstGroup
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:gravity="center"
tools:context=".SecondActivity">
<com.touch.harvic.testtouchevent.CustomSecondGroup
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="#00ff00"
android:padding="50dp">
<com.touch.harvic.testtouchevent.CustomTextView
android:id="@+id/tv"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:padding="10dp"
android:background="#ff0000"
android:text="Text View"/>
</com.touch.harvic.testtouchevent.CustomSecondGroup