前两天朋友问我一个问题,如何实现从手机系统相册加载一张图片,实现拖拽、缩放功能并且能镜像显示,当时我的第一反应是继承ImageView,重写onDraw方法,下面就按照这个思路实现下需求。
打开系统相册采用隐式Intent,因为调用者需要接受返回的结果,所以使用startActivityForResult启动Activity,打开系统相册核心代码如下
public final static int ALBUM_PIC_CODE = 1; ... Intent intent = new Intent(Intent.ACTION_PICK); intent.setData(MediaStore.Images.Media.EXTERNAL_CONTENT_URI); startActivityForResult(intent, ALBUM_PIC_CODE);
通过打印日志的方式看一下从系统相册中选择图片以后返回的内容是什么:
可以断定这是个Uri,那么好,我们直接通过Uri获取图片即可:
protected void onActivityResult(int requestCode, int resultCode, Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode == ALBUM_PIC_CODE && resultCode == RESULT_OK){ Log.i("onActivityResult", data.getData().toString()); try { Uri imageUri = data.getData(); Bitmap bitmap= BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri)); iv.setImageBitmap(bitmap); } catch (FileNotFoundException e) { e.printStackTrace(); } } }
这样我们就通过Uri以流的方式载入了图片
载入图片以后,我们需要在显示以前对图片进行处理以达到所需的显示效果,我了方便我就直接对系统控件ImageView进行重写来达到想要的功能。
要实现拖拽功能,首先会想到重写onTouchEvent方法,需要在手指落下时获取坐标点的信息,并在手指移动时根据移动的位置实时更新图片位置.
public boolean onTouchEvent(MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: lastX = event.getX(); lastY = event.getY(); break; case MotionEvent.ACTION_MOVE: float currentX = event.getX(); float currentY = event.getY(); offsetX += currentX - lastX;//计算移动距离 offsetY += currentY - lastY; lastX = currentX;//替换上一次位置 lastY = currentY; break; case MotionEvent.ACTION_UP: break; default: break; } super.onTouchEvent(event); return true; }
重写onDraw方法对图片位置进行实时更新
protected void onDraw(Canvas canvas) { if (bitmap != null) { bitmapWidth = bitmap.getWidth(); bitmapHeight = bitmap.getHeight(); try { int scaleHeight = (int) (viewWidth / (float) bitmapWidth * bitmapHeight); bitmapLeft = (int) offsetX; bitmapRight = (int) (viewWidth + offsetX); bitmapTop = (int) offsetY; bitmapBottom = (int) (scaleHeight + offsetY); Paint paint = new Paint(); canvas.drawBitmap(bitmap, new Rect(0, 0, bitmapWidth, bitmapHeight), new Rect(bitmapLeft, bitmapTop, bitmapRight, bitmapBottom), paint); } catch (Exception e) { e.printStackTrace(); } invalidate(); } else { try { bitmap = ((BitmapDrawable) getDrawable()).getBitmap(); } catch (Exception e) { e.printStackTrace(); bitmap = null; } } }