图片缩放时java.lang.IllegalArgumentException: pointerIndex out of range解决方案

本文详细介绍了使用多点触控实现图片放大缩小功能的代码实现,并针对可能出现的错误进行了深入分析,特别关注了手指操作过程中可能导致的越界问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近在学习多点触控Multi-Touch照着别人的代码写了一个多点触控的Demo,功能是实现图片的放大缩小。

一、实现的代码,下面是可能会出现错误的代码:

 

package com.example.webproject;

import android.app.Activity;
import android.graphics.Matrix;
import android.graphics.PointF;
import android.os.Bundle;
import android.util.FloatMath;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnTouchListener;
import android.widget.ImageView;

public class ImageMarixActivity extends Activity{
	private static final String TAG = "ImageMarixActivity";
	private static final int RESULT_CODE_NOFOUND = 200;
	private ImageView matrixImageView;
	

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		// TODO Auto-generated method stub
		super.onCreate(savedInstanceState);
		setContentView(R.layout.imagemarix_activity);
		matrixImageView = (ImageView) findViewById(R.id.matriximageview);
		//监听图片的触摸事件
		matrixImageView.setOnTouchListener(new TouchListener());
	}
	
	
	public final class TouchListener implements OnTouchListener{
		//记录拖拽是放大还是缩小
		private int mode = 0;//初始状态
		private int DRAG_MODE = 1;//拖拽模式
		private int ZOOM_MODE = 2 ;//缩小模式
		//记录开始的位置
		private PointF startPoint = new PointF();
		//两个手指中间的距离
		private PointF midPont;
		//拖拉图片开始移动的坐标位置
		private Matrix matrix = new Matrix();
		//拖拉图片当前移动的坐标位置
		private Matrix currentMatrix = new Matrix();
		//开始两个手指的距离
		private float twopointstartDis;
		@Override
		public boolean onTouch(View v, MotionEvent event) {
			// TODO Auto-generated method stub
			switch (event.getAction()& MotionEvent.ACTION_MASK) {
			//手指按下的时候
			case MotionEvent.ACTION_DOWN:
				mode = DRAG_MODE;
				//记录当前的位置
				Log.i(TAG, "当前的位置"+matrixImageView.getImageMatrix());
				currentMatrix.set(matrixImageView.getImageMatrix());
				//记录手指按下去的位置
				startPoint.set(event.getX(), event.getY());
				break;
				
			case MotionEvent.ACTION_MOVE:
				Log.i(TAG, "移动"+"模式"+mode);
				//如果是拖拉模式
				if(mode == DRAG_MODE)
				{
					float dx = event.getX() - startPoint.x;
					float dy = event.getY() - startPoint.y;
					matrix.set(currentMatrix);//在原有的基础上移动
					matrix.postTranslate(dx, dy);
				}
				else if(mode == ZOOM_MODE)
				{
					Log.i(TAG, "ZOOM_MODE获取第0个x"+event.getX(0));
					Log.i(TAG, "获取第一个x"+event.getX(1));
					float endDis = distance(event);
					if(endDis >10f)
					{
						matrix.set(currentMatrix);
						float scale =endDis / twopointstartDis;
						matrix.postScale(scale, scale,midPont.x,midPont.y);
					}
				}
				break;
				
			case MotionEvent.ACTION_UP:
				Log.i(TAG, "一只手指点击后弹起来");
//				mode =0;
				break;
				
				
			case MotionEvent.ACTION_POINTER_DOWN:
				mode = ZOOM_MODE;
				twopointstartDis = distance(event);
				if(twopointstartDis>10f)
				{
					midPont = mid(event);
					currentMatrix.set(matrixImageView.getImageMatrix());
				}
				break;
				
			case MotionEvent.ACTION_POINTER_UP:
				Log.i(TAG, "两只手指点击后弹起来");
//				Log.i(TAG, "获取第一个x"+event.getX(1)+"获取的第一个x"+event.getY(0));
//				Log.i(TAG, "获取第一个x"+(event.getX(1)-event.getY(0)));
//				<span style="color:#ff0000;">mode = 0;//解决办法是在up的时候保证将模式置为0,在move事件的时候处于放大缩小模式,就不会getX(1)</span>
				break;

			default:
				break;
			}
			matrixImageView.setImageMatrix(matrix);
			return true;
		}
		
	}
	
	public float distance(MotionEvent event)
	{
		Log.i(TAG, "调用测距");
		float dx = event.getX(1) - event.getX(0);
		float dy = event.getY(1) - event.getY(0);
		return FloatMath.sqrt(dx*dx+dy*dy);
	}
	
	public PointF mid(MotionEvent event)
	{
		float dx = (event.getX(1)+event.getX(0))/2;
		float dy = (event.getY(1)+event.getY(0))/2;
		return new PointF(dx,dy);
	}

}


出现的错误:

 

 

自己研究了好一阵子才发现与你手指触控点有关系,当你图片放大缩小的时候,手指抬起来触发MotionEvent.ACTION_POINTER_UP的事件,当一个手指抬起来产生了move事件,继续去触发MotionEvent.ACTION_MOVE事件(跟你手指操作情况有关系),如果你的move事件继续调用getX(1),因为已经没有第二个触控点了,你继续getX(1)就会报触控点越界。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值