Android 自定义View实现单击和双击事件的方法
作者:jingxian 发布时间:2022-03-23 19:21:18
标签:自定义,View,单击,双击
自定义View,
1. 自定义一个Runnable线程TouchEventCountThread , 用来统计500ms内的点击次数
2. 在MyView中的 onTouchEvent 中调用 上面的线程
3. 自定义一个Handler, 在TouchEventHandler 中 处理 统计到的点击事件, 单击, 双击, 三击, 都可以处理
核心代码如下:
public class MyView extends View {
......
// 统计500ms内的点击次数
TouchEventCountThread mInTouchEventCount = new TouchEventCountThread();
// 根据TouchEventCountThread统计到的点击次数, perform单击还是双击事件
TouchEventHandler mTouchEventHandler = new TouchEventHandler();
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
if (0 == mInTouchEventCount.touchCount) // 第一次按下时,开始统计
postDelayed(mInTouchEventCount, 500);
break;
case MotionEvent.ACTION_UP:
// 一次点击事件要有按下和抬起, 有抬起必有按下, 所以只需要在ACTION_UP中处理
mInTouchEventCount.touchCount++;
// 如果是长按操作, 则Handler的消息,不能将touchCount置0, 需要特殊处理
if(mInTouchEventCount.isLongClick) {
mInTouchEventCount.touchCount = 0;
mInTouchEventCount.isLongClick = false;
}
break;
case MotionEvent.ACTION_MOVE:
break;
case MotionEvent.ACTION_CANCEL:
break;
default:
break;
}
return super.onTouchEvent(event);
}
public class TouchEventCountThread implements Runnable {
public int touchCount = 0;
public boolean isLongClick = false;
@Override
public void run() {
Message msg = new Message();
if(0 == touchCount){ // long click
isLongClick = true;
} else {
msg.arg1 = touchCount;
mTouchEventHandler.sendMessage(msg);
touchCount = 0;
}
}
}
public class TouchEventHandler extends Handler {
@Override
public void handleMessage(Message msg) {
Toast.makeText(mContext, "touch " + msg.arg1 + " time.", Toast.LENGTH_SHORT).show();
}
}
......
}
包装以后如下, 这样就能在别的地方调用了:
public interface OnDoubleClickListener{
void onDoubleClick(View v);
}
private OnDoubleClickListener mOnDoubleClickListener;
public void setOnDoubleClickListener(MyView.OnDoubleClickListener l) {
mOnDoubleClickListener = l;
}
public boolean performDoubleClick() {
boolean result = false;
if(mOnDoubleClickListener != null) {
mOnDoubleClickListener.onDoubleClick(this);
result = true;
}
return result;
}
public class TouchEventHandler extends Handler {
@Override
public void handleMessage(Message msg) {
if(2 == msg.arg1)
performDoubleClick();
}
}
在Activity中使用:
myView1.setOnDoubleClickListener(new MyView.OnDoubleClickListener() {
@Override
public void onDoubleClick(View v) {
Toast.makeText(mContext,"double click", Toast.LENGTH_SHORT).show();
}
});
全部代码
MyView.java
package com.carloz.test.myapplication.view;
import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.os.Handler;
import android.os.Message;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.Toast;
import com.carloz.test.myapplication.R;
/**
* Created by root on 15-11-9.
*/
public class MyView extends View {
private Paint mPaint = new Paint();
private boolean mNotDestroy = true;
private int mCount = 0;
private MyThread myThread;
Bitmap bitmap;
// attrs
private String mText;
private boolean mStartChange;
Context mContext;
public MyView(Context context) {
super(context);
init();
}
public MyView(Context context, AttributeSet attrs) {
super(context, attrs);
TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.MyView);
mText = ta.getString(R.styleable.MyView_text);
mStartChange = ta.getBoolean(R.styleable.MyView_startChange, false);
// Log.d("ASDF", "mText=" + mText + ", mStartChange=" + mStartChange);
ta.recycle();
init();
}
@Override
protected void onFinishInflate() {
super.onFinishInflate();
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
super.onLayout(changed, left, top, right, bottom);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
mPaint.setTextSize(50);
canvas.drawText(mText + mCount++, 20f, 100f, mPaint);
canvas.save();
canvas.rotate(60, getWidth() / 2, getHeight() / 2);
canvas.drawBitmap(bitmap, 20f, 50f, mPaint);
canvas.restore();
if (null == myThread) {
myThread = new MyThread();
myThread.start();
}
}
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
return super.dispatchTouchEvent(ev);
}
@Override
protected void onAttachedToWindow() {
super.onAttachedToWindow();
mNotDestroy = true;
}
@Override
protected void onDetachedFromWindow() {
mNotDestroy = false;
super.onDetachedFromWindow();
}
// 统计500ms内的点击次数
TouchEventCountThread mInTouchEventCount = new TouchEventCountThread();
// 根据TouchEventCountThread统计到的点击次数, perform单击还是双击事件
TouchEventHandler mTouchEventHandler = new TouchEventHandler();
@Override
public boolean onTouchEvent(MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
if (0 == mInTouchEventCount.touchCount) // 第一次按下时,开始统计
postDelayed(mInTouchEventCount, 500);
break;
case MotionEvent.ACTION_UP:
// 一次点击事件要有按下和抬起, 有抬起必有按下, 所以只需要在ACTION_UP中处理
mInTouchEventCount.touchCount++;
// 如果是长按操作, 则Handler的消息,不能将touchCount置0, 需要特殊处理
if(mInTouchEventCount.isLongClick) {
mInTouchEventCount.touchCount = 0;
mInTouchEventCount.isLongClick = false;
}
break;
case MotionEvent.ACTION_MOVE:
break;
case MotionEvent.ACTION_CANCEL:
break;
default:
break;
}
return super.onTouchEvent(event);
}
public class TouchEventCountThread implements Runnable {
public int touchCount = 0;
public boolean isLongClick = false;
@Override
public void run() {
Message msg = new Message();
if(0 == touchCount){ // long click
isLongClick = true;
} else {
msg.arg1 = touchCount;
mTouchEventHandler.sendMessage(msg);
touchCount = 0;
}
}
}
public class TouchEventHandler extends Handler {
@Override
public void handleMessage(Message msg) {
Toast.makeText(mContext, "touch " + msg.arg1 + " time.", Toast.LENGTH_SHORT).show();
}
}
class MyThread extends Thread {
@Override
public void run() {
super.run();
while (mNotDestroy) {
if (mStartChange) {
postInvalidate();
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
}
public void init() {
mContext = getContext();
bitmap = BitmapFactory.decodeResource(getResources(), R.drawable.ic_launcher);
}
public void setText(String mText) {
this.mText = mText;
}
public void setStartChange(boolean mStartChange) {
this.mStartChange = mStartChange;
}
public boolean getStartChange() {
return this.mStartChange;
}
}
attrs.xml
<?xml version="1.0" encoding="utf-8"?>
<resources>
<declare-styleable name="MyView">
<attr name="text" format="string"/>
<attr name="startChange" format="boolean"/>
</declare-styleable>
</resources>
postDelayed方法最终是靠 Handler 的 postDelayed 方法 实现原理如下
public final boolean postDelayed(Runnable r, long delayMillis)
{
return sendMessageDelayed(getPostMessage(r), delayMillis);
}
public final boolean sendMessageDelayed(Message msg, long delayMillis)
{
if (delayMillis < 0) {
delayMillis = 0;
}
return sendMessageAtTime(msg, SystemClock.uptimeMillis() + delayMillis);
}
public boolean sendMessageAtTime(Message msg, long uptimeMillis) {
MessageQueue queue = mQueue;
if (queue == null) {
RuntimeException e = new RuntimeException(
this + " sendMessageAtTime() called with no mQueue");
Log.w("Looper", e.getMessage(), e);
return false;
}
return enqueueMessage(queue, msg, uptimeMillis); // 然后在MessageQueue中会比较时间顺序
}


猜你喜欢
- if语句使用布尔表达式或布尔值作为分支条件来进行分支控制,其中if语句有如下三种形式:第一种形式:if ( logic expression
- 这篇文章主要介绍了SpringBoot实现 * 、过滤器、 * 过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考
- Mybatis-plus官网地址:https://baomidou.com/配置mysql在配置文件连接mysqlspring.dataso
- 问题描述提交表单到servlet时出现空白页面,但又网站不报错截图如下遇到这个问题查找了很多,仍没有解决,刚开始接触IDEA,以为是路径配置
- 本文实例展示了DevExpress获取TreeList可视区域节点集合的实现方法,是比较有实用价值的技巧。分享给大家供大家参考。具体实现方法
- 最近我要做一个爬虫。这个爬虫需要如下几个步骤:1 填写注册内容(需要邮箱注册)2 过拖拽验证码(geetest)3 注册成功会给邮箱发一封确
- using System;using System.Collections.Generic;using System.Linq;using
- 1.break 关键字break 主要用在:① 循环语句中② switch语句中它的作用:用来跳出整个语句块。1.1语法break 的用法很
- 背景原生的TextView是支持跑马灯效果的,但是在项目中实际用了之后,达不到需求,原因是内容滚动太慢,速度无法调节。因此,需要自定义一个可
- 简单工厂简单工厂模式是属于创建型模式,是工厂模式的一种。简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例。定义了一个创建对象的类,由
- 今天是圣诞节平安夜,为此特别制作了一个雪花飘落的场景,我们的雪花渲染方式不同于网上流行的使用Camera Filter,需要将脚本挂接到相机
- 过滤掉其他的播放器,使用我自己的播放器来做 wv.set
- Runtime.getRuntime().exec 路径包含空格1. 现象java代码通过Runtime.getRuntime().exec
- 本文实例讲述了Java创建ZIP压缩文件的方法。分享给大家供大家参考。具体如下:这里注意:建议使用org.apache.tools.zip.
- 本文实例讲述了C#基于HttpWebRequest实现发送HTTP请求的方法。分享给大家供大家参考,具体如下:调用第三方API的时候要用到H
- 查看JDK1.8 ArrayList的源代码1、默认初始容量为10 /** * Default i
- 起因unity程序build到pc上,拿到其他人的机器上结果有些功能不正常,看log里面大概是Fallback handler could
- 一、需求分析最近公司项目要实现一个需求要满足以下功能: 1)显示一个 list
- 本文实例为大家分享了Android实现背景图滑动变大松开回弹的具体代码,供大家参考,具体内容如下原图放大后1、自定义view继承Scroll
- 已经有很多关于 Flutter WebView 的文章了,为什么还要写一篇。两个原因:Flutter WebView 是 Flutter 开