Android多点触控技术实战,自由地对图片进行缩放和移动

转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/11100327


在上一篇文章中我带着大家一起实现了Android瀑布流照片墙的效果,虽然这种效果很炫很酷,但其实还只能算是一个半成品,因为照片墙中所有的图片都是只能看不能点的。因此本篇文章中,我们就来对这一功能进行完善,加入点击图片就能浏览大图的功能,并且在浏览大图的时候还可以通过多点触控的方式对图片进行缩放。


如果你还没有看过 Android瀑布流照片墙实现,体验不规则排列的美感 这篇文章,请尽量先去阅读完再来看本篇文章,因为这次的代码完全是在上次的基础上进行开发的。


那我们现在就开始动手吧,首先打开上次的PhotoWallFallsDemo项目,在里面加入一个ZoomImageView类,这个类就是用于进行大图展示和多点触控缩放的,代码如下所示:

  1. public class ZoomImageView extends View {
  2. /**
  3. * 初始化状态常量
  4. */
  5. public static final int STATUS_INIT = 1;
  6. /**
  7. * 图片放大状态常量
  8. */
  9. public static final int STATUS_ZOOM_OUT = 2;
  10. /**
  11. * 图片缩小状态常量
  12. */
  13. public static final int STATUS_ZOOM_IN = 3;
  14. /**
  15. * 图片拖动状态常量
  16. */
  17. public static final int STATUS_MOVE = 4;
  18. /**
  19. * 用于对图片进行移动和缩放变换的矩阵
  20. */
  21. private Matrix matrix = new Matrix();
  22. /**
  23. * 待展示的Bitmap对象
  24. */
  25. private Bitmap sourceBitmap;
  26. /**
  27. * 记录当前操作的状态,可选值为STATUS_INIT、STATUS_ZOOM_OUT、STATUS_ZOOM_IN和STATUS_MOVE
  28. */
  29. private int currentStatus;
  30. /**
  31. * ZoomImageView控件的宽度
  32. */
  33. private int width;
  34. /**
  35. * ZoomImageView控件的高度
  36. */
  37. private int height;
  38. /**
  39. * 记录两指同时放在屏幕上时,中心点的横坐标值
  40. */
  41. private float centerPointX;
  42. /**
  43. * 记录两指同时放在屏幕上时,中心点的纵坐标值
  44. */
  45. private float centerPointY;
  46. /**
  47. * 记录当前图片的宽度,图片被缩放时,这个值会一起变动
  48. */
  49. private float currentBitmapWidth;
  50. /**
  51. * 记录当前图片的高度,图片被缩放时,这个值会一起变动
  52. */
  53. private float currentBitmapHeight;
  54. /**
  55. * 记录上次手指移动时的横坐标
  56. */
  57. private float lastXMove = - 1;
  58. /**
  59. * 记录上次手指移动时的纵坐标
  60. */
  61. private float lastYMove = - 1;
  62. /**
  63. * 记录手指在横坐标方向上的移动距离
  64. */
  65. private float movedDistanceX;
  66. /**
  67. * 记录手指在纵坐标方向上的移动距离
  68. */
  69. private float movedDistanceY;
  70. /**
  71. * 记录图片在矩阵上的横向偏移值
  72. */
  73. private float totalTranslateX;
  74. /**
  75. * 记录图片在矩阵上的纵向偏移值
  76. */
  77. private float totalTranslateY;
  78. /**
  79. * 记录图片在矩阵上的总缩放比例
  80. */
  81. private float totalRatio;
  82. /**
  83. * 记录手指移动的距离所造成的缩放比例
  84. */
  85. private float scaledRatio;
  86. /**
  87. * 记录图片初始化时的缩放比例
  88. */
  89. private float initRatio;
  90. /**
  91. * 记录上次两指之间的距离
  92. */
  93. private double lastFingerDis;
  94. /**
  95. * ZoomImageView构造函数,将当前操作状态设为STATUS_INIT。
  96. *
  97. * @param context
  98. * @param attrs
  99. */
  100. public ZoomImageView(Context context, AttributeSet attrs) {
  101. super(context, attrs);
  102. currentStatus = STATUS_INIT;
  103. }
  104. /**
  105. * 将待展示的图片设置进来。
  106. *
  107. * @param bitmap
  108. * 待展示的Bitmap对象
  109. */
  110. public void setImageBitmap(Bitmap bitmap) {
  111. sourceBitmap = bitmap;
  112. invalidate();
  113. }
  114. @Override
  115. protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
  116. super.onLayout(changed, left, top, right, bottom);
  117. if (changed) {
  118. // 分别获取到ZoomImageView的宽度和高度
  119. width = getWidth();
  120. height = getHeight();
  121. }
  122. }
  123. @Override
  124. public boolean onTouchEvent(MotionEvent event) {
  125. switch (event.getActionMasked()) {
  126. case MotionEvent.ACTION_POINTER_DOWN:
  127. if (event.getPointerCount() == 2) {
  128. // 当有两个手指按在屏幕上时,计算两指之间的距离
  129. lastFingerDis = distanceBetweenFingers(event);
  130. }
  131. break;
  132. case MotionEvent.ACTION_MOVE:
  133. if (event.getPointerCount() == 1) {
  134. // 只有单指按在屏幕上移动时,为拖动状态
  135. float xMove = event.getX();
  136. float yMove = event.getY();
  137. if (lastXMove == - 1 && lastYMove == - 1) {
  138. lastXMove = xMove;
  139. lastYMove = yMove;
  140. }
  141. currentStatus = STATUS_MOVE;
  142. movedDistanceX = xMove - lastXMove;
  143. movedDistanceY = yMove - lastYMove;
  144. // 进行边界检查,不允许将图片拖出边界
  145. if (totalTranslateX + movedDistanceX > 0) {
  146. movedDistanceX = 0;
  147. } else if (width - (totalTranslateX + movedDistanceX) > currentBitmapWidth) {
  148. movedDistanceX = 0;
  149. }
  150. if (totalTranslateY + movedDistanceY > 0) {
  151. movedDistanceY = 0;
  152. } else if (height - (totalTranslateY + movedDistanceY) > currentBitmapHeight) {
  153. movedDistanceY = 0;
  154. }
  155. // 调用onDraw()方法绘制图片
  156. invalidate();
  157. lastXMove = xMove;
  158. lastYMove = yMove;
  159. } else if (event.getPointerCount() == 2) {
  160. // 有两个手指按在屏幕上移动时,为缩放状态
  161. centerPointBetweenFingers(event);
  162. double fingerDis = distanceBetweenFingers(event);
  163. if (fingerDis > lastFingerDis) {
  164. currentStatus = STATUS_ZOOM_OUT;
  165. } else {
  166. currentStatus = STATUS_ZOOM_IN;
  167. }
  168. // 进行缩放倍数检查,最大只允许将图片放大4倍,最小可以缩小到初始化比例
  169. if ((currentStatus == STATUS_ZOOM_OUT && totalRatio < 4 * initRatio)
  170. || (currentStatus == STATUS_ZOOM_IN && totalRatio > initRatio)) {
  171. scaledRatio = ( float) (fingerDis / lastFingerDis);
  172. totalRatio = totalRatio * scaledRatio;
  173. if (totalRatio > 4 * initRatio) {
  174. totalRatio = 4 * initRatio;
  175. } else if (totalRatio < initRatio) {
  176. totalRatio = initRatio;
  177. }
  178. // 调用onDraw()方法绘制图片
  179. invalidate();
  180. lastFingerDis = fingerDis;
  181. }
  182. }
  183. break;
  184. case MotionEvent.ACTION_POINTER_UP:
  185. if (event.getPointerCount() == 2) {
  186. // 手指离开屏幕时将临时值还原
  187. lastXMove = - 1;
  188. lastYMove = - 1;
  189. }
  190. break;
  191. case MotionEvent.ACTION_UP:
  192. // 手指离开屏幕时将临时值还原
  193. lastXMove = - 1;
  194. lastYMove = - 1;
  195. break;
  196. default:
  197. break;
  198. }
  199. return true;
  200. }
  201. /**
  202. * 根据currentStatus的值来决定对图片进行什么样的绘制操作。
  203. */
  204. @Override
  205. protected void onDraw(Canvas canvas) {
  206. super.onDraw(canvas);
  207. switch (currentStatus) {
  208. case STATUS_ZOOM_OUT:
  209. case STATUS_ZOOM_IN:
  210. zoom(canvas);
  211. break;
  212. case STATUS_MOVE:
  213. move(canvas);
  214. break;
  215. case STATUS_INIT:
  216. initBitmap(canvas);
  217. default:
  218. canvas.drawBitmap(sourceBitmap, matrix, null);
  219. break;
  220. }
  221. }
  222. /**
  223. * 对图片进行缩放处理。
  224. *
  225. * @param canvas
  226. */
  227. private void zoom(Canvas canvas) {
  228. matrix.reset();
  229. // 将图片按总缩放比例进行缩放
  230. matrix.postScale(totalRatio, totalRatio);
  231. float scaledWidth = sourceBitmap.getWidth() * totalRatio;
  232. float scaledHeight = sourceBitmap.getHeight() * totalRatio;
  233. float translateX = 0f;
  234. float translateY = 0f;
  235. // 如果当前图片宽度小于屏幕宽度,则按屏幕中心的横坐标进行水平缩放。否则按两指的中心点的横坐标进行水平缩放
  236. if (currentBitmapWidth < width) {
  237. translateX = (width - scaledWidth) / 2f;
  238. } else {
  239. translateX = totalTranslateX * scaledRatio + centerPointX * ( 1 - scaledRatio);
  240. // 进行边界检查,保证图片缩放后在水平方向上不会偏移出屏幕
  241. if (translateX > 0) {
  242. translateX = 0;
  243. } else if (width - translateX > scaledWidth) {
  244. translateX = width - scaledWidth;
  245. }
  246. }
  247. // 如果当前图片高度小于屏幕高度,则按屏幕中心的纵坐标进行垂直缩放。否则按两指的中心点的纵坐标进行垂直缩放
  248. if (currentBitmapHeight < height) {
  249. translateY = (height - scaledHeight) / 2f;
  250. } else {
  251. translateY = totalTranslateY * scaledRatio + centerPointY * ( 1 - scaledRatio);
  252. // 进行边界检查,保证图片缩放后在垂直方向上不会偏移出屏幕
  253. if (translateY > 0) {
  254. translateY = 0;
  255. } else if (height - translateY > scaledHeight) {
  256. translateY = height - scaledHeight;
  257. }
  258. }
  259. // 缩放后对图片进行偏移,以保证缩放后中心点位置不变
  260. matrix.postTranslate(translateX, translateY);
  261. totalTranslateX = translateX;
  262. totalTranslateY = translateY;
  263. currentBitmapWidth = scaledWidth;
  264. currentBitmapHeight = scaledHeight;
  265. canvas.drawBitmap(sourceBitmap, matrix, null);
  266. }
  267. /**
  268. * 对图片进行平移处理
  269. *
  270. * @param canvas
  271. */
  272. private void move(Canvas canvas) {
  273. matrix.reset();
  274. // 根据手指移动的距离计算出总偏移值
  275. float translateX = totalTranslateX + movedDistanceX;
  276. float translateY = totalTranslateY + movedDistanceY;
  277. // 先按照已有的缩放比例对图片进行缩放
  278. matrix.postScale(totalRatio, totalRatio);
  279. // 再根据移动距离进行偏移
  280. matrix.postTranslate(translateX, translateY);
  281. totalTranslateX = translateX;
  282. totalTranslateY = translateY;
  283. canvas.drawBitmap(sourceBitmap, matrix, null);
  284. }
  285. /**
  286. * 对图片进行初始化操作,包括让图片居中,以及当图片大于屏幕宽高时对图片进行压缩。
  287. *
  288. * @param canvas
  289. */
  290. private void initBitmap(Canvas canvas) {
  291. if (sourceBitmap != null) {
  292. matrix.reset();
  293. int bitmapWidth = sourceBitmap.getWidth();
  294. int bitmapHeight = sourceBitmap.getHeight();
  295. if (bitmapWidth > width || bitmapHeight > height) {
  296. if (bitmapWidth - width > bitmapHeight - height) {
  297. // 当图片宽度大于屏幕宽度时,将图片等比例压缩,使它可以完全显示出来
  298. float ratio = width / (bitmapWidth * 1.0f);
  299. matrix.postScale(ratio, ratio);
  300. float translateY = (height - (bitmapHeight * ratio)) / 2f;
  301. // 在纵坐标方向上进行偏移,以保证图片居中显示
  302. matrix.postTranslate( 0, translateY);
  303. totalTranslateY = translateY;
  304. totalRatio = initRatio = ratio;
  305. } else {
  306. // 当图片高度大于屏幕高度时,将图片等比例压缩,使它可以完全显示出来
  307. float ratio = height / (bitmapHeight * 1.0f);
  308. matrix.postScale(ratio, ratio);
  309. float translateX = (width - (bitmapWidth * ratio)) / 2f;
  310. // 在横坐标方向上进行偏移,以保证图片居中显示
  311. matrix.postTranslate(translateX, 0);
  312. totalTranslateX = translateX;
  313. totalRatio = initRatio = ratio;
  314. }
  315. currentBitmapWidth = bitmapWidth * initRatio;
  316. currentBitmapHeight = bitmapHeight * initRatio;
  317. } else {
  318. // 当图片的宽高都小于屏幕宽高时,直接让图片居中显示
  319. float translateX = (width - sourceBitmap.getWidth()) / 2f;
  320. float translateY = (height - sourceBitmap.getHeight()) / 2f;
  321. matrix.postTranslate(translateX, translateY);
  322. totalTranslateX = translateX;
  323. totalTranslateY = translateY;
  324. totalRatio = initRatio = 1f;
  325. currentBitmapWidth = bitmapWidth;
  326. currentBitmapHeight = bitmapHeight;
  327. }
  328. canvas.drawBitmap(sourceBitmap, matrix, null);
  329. }
  330. }
  331. /**
  332. * 计算两个手指之间的距离。
  333. *
  334. * @param event
  335. * @return 两个手指之间的距离
  336. */
  337. private double distanceBetweenFingers(MotionEvent event) {
  338. float disX = Math.abs(event.getX( 0) - event.getX( 1));
  339. float disY = Math.abs(event.getY( 0) - event.getY( 1));
  340. return Math.sqrt(disX * disX + disY * disY);
  341. }
  342. /**
  343. * 计算两个手指之间中心点的坐标。
  344. *
  345. * @param event
  346. */
  347. private void centerPointBetweenFingers(MotionEvent event) {
  348. float xPoint0 = event.getX( 0);
  349. float yPoint0 = event.getY( 0);
  350. float xPoint1 = event.getX( 1);
  351. float yPoint1 = event.getY( 1);
  352. centerPointX = (xPoint0 + xPoint1) / 2;
  353. centerPointY = (yPoint0 + yPoint1) / 2;
  354. }
  355. }

由于这个类是整个多点触控缩放功能最核心的一个类,我在这里给大家详细的讲解一下。首先在ZoomImageView里我们定义了四种状态,STATUS_INIT、STATUS_ZOOM_OUT、STATUS_ZOOM_IN和STATUS_MOVE,这四个状态分别代表初始化、放大、缩小和移动这几个动作,然后在构造函数里我们将当前状态置为初始化状态。接着我们可以调用setImageBitmap()方法把要显示的图片对象传进去,这个方法会invalidate一下当前的View,因此onDraw()方法就会得到执行。然后在onDraw()方法里判断出当前的状态是初始化状态,所以就会调用initBitmap()方法进行初始化操作。


那我们就来看一下initBitmap()方法,在这个方法中首先对图片的大小进行了判断,如果图片的宽和高都是小于屏幕的宽和高的,则直接将这张图片进行偏移,让它能够居中显示在屏幕上。如果图片的宽度大于屏幕的宽度,或者图片的高度大于屏幕的高度,则将图片进行等比例压缩,让图片的的宽或高正好等同于屏幕的宽或高,保证在初始化状态下图片一定能完整地显示出来。这里所有的偏移和缩放操作都是通过矩阵来完成的,我们把要缩放和偏移的值都存放在矩阵中,然后在绘制图片的时候传入这个矩阵对象就可以了。


图片初始化完成之后,就可以对图片进行缩放处理了。这里在onTouchEvent()方法来对点击事件进行判断,如果发现有两个手指同时按在屏幕上(使用event.getPointerCount()判断)就将当前状态置为缩放状态,并调用distanceBetweenFingers()来得到两指之间的距离,以计算出缩放比例。然后invalidate一下,就会在onDraw()方法中就会调用zoom()方法。之后就在这个方法里根据当前的缩放比例以及中心点的位置对图片进行缩放和偏移,具体的逻辑大家请仔细阅读代码,注释已经写得非常清楚。


然后当只有一个手指按在屏幕上时,就把当前状态置为移动状态,之后会对手指的移动距离进行计算,并处理了边界检查的工作,以防止图片偏移出屏幕。然后invalidate一下当前的view,又会进入到onDraw()方法中,这里判断出当前是移动状态,于是会调用move()方法。move()方法中的代码非常简单,就是根据手指移动的距离对图片进行偏移就可以了。


介绍完了ZoomImageView,然后我们新建一个布局image_details.xml,在布局中直接引用创建好的ZoomImageView:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <com.example.photowallfallsdemo.ZoomImageView xmlns:android="http://schemas.android.com/apk/res/android"
  3. android:id= "@+id/zoom_image_view"
  4. android:layout_width= "match_parent"
  5. android:layout_height= "match_parent"
  6. android:background= "#000000" >
  7. </com.example.photowallfallsdemo.ZoomImageView>
接着创建一个Activity,在这个Activity中来加载image_details布局。新建ImageDetailsActivity,代码如下所示:
  1. public class ImageDetailsActivity extends Activity {
  2. private ZoomImageView zoomImageView;
  3. @Override
  4. protected void onCreate(Bundle savedInstanceState) {
  5. super.onCreate(savedInstanceState);
  6. requestWindowFeature(Window.FEATURE_NO_TITLE);
  7. setContentView(R.layout.image_details);
  8. zoomImageView = (ZoomImageView) findViewById(R.id.zoom_image_view);
  9. String imagePath = getIntent().getStringExtra( "image_path");
  10. Bitmap bitmap = BitmapFactory.decodeFile(imagePath);
  11. zoomImageView.setImageBitmap(bitmap);
  12. }
  13. }

可以看到,首先我们获取到了ZoomImageView的实例,然后又通过Intent得到了需要展示的图片路径,接着使用BitmapFactory将路径下的图片加载到内存中,然后调用ZoomImageView的setImageBitmap()方法将图片传入,就可以让这张图片展示出来了。


接下来我们需要考虑的,就是如何在照片墙上给图片增加点击事件,让它能够启动ImageDetailsActivity了。其实这也很简单,只需要在动态添加图片的时候给每个ImageView的实例注册一下点击事件就好了,修改MyScrollView中addImage()方法的代码,如下所示:

  1. private void addImage(Bitmap bitmap, int imageWidth, int imageHeight) {
  2. LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(imageWidth,
  3. imageHeight);
  4. if (mImageView != null) {
  5. mImageView.setImageBitmap(bitmap);
  6. } else {
  7. ImageView imageView = new ImageView(getContext());
  8. imageView.setLayoutParams(params);
  9. imageView.setImageBitmap(bitmap);
  10. imageView.setScaleType(ScaleType.FIT_XY);
  11. imageView.setPadding( 5, 5, 5, 5);
  12. imageView.setTag(R.string.image_url, mImageUrl);
  13. imageView.setOnClickListener( new OnClickListener() {
  14. @Override
  15. public void onClick(View v) {
  16. Intent intent = new Intent(getContext(), ImageDetailsActivity.class);
  17. intent.putExtra( "image_path", getImagePath(mImageUrl));
  18. getContext().startActivity(intent);
  19. }
  20. });
  21. findColumnToAdd(imageView, imageHeight).addView(imageView);
  22. imageViewList.add(imageView);
  23. }
  24. }

可以看到,这里我们调用了ImageView的setOnClickListener()方法来给图片增加点击事件,当用户点击了照片墙中的任意图片时,就会启动ImageDetailsActivity,并将图片的路径传递过去。


由于我们添加了一个新的Activity,别忘了在AndroidManifest.xml文件里注册一下:

  1. <?xml version="1.0" encoding="utf-8"?>
  2. <manifest xmlns:android="http://schemas.android.com/apk/res/android"
  3. package= "com.example.photowallfallsdemo"
  4. android:versionCode= "1"
  5. android:versionName= "1.0" >
  6. <uses-sdk
  7. android:minSdkVersion= "14"
  8. android:targetSdkVersion= "17" />
  9. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
  10. <uses-permission android:name="android.permission.INTERNET" />
  11. <application
  12. android:allowBackup= "true"
  13. android:icon= "@drawable/ic_launcher"
  14. android:label= "@string/app_name"
  15. android:theme= "@style/AppTheme" >
  16. <activity
  17. android:name= "com.example.photowallfallsdemo.MainActivity"
  18. android:label= "@string/app_name" >
  19. <intent-filter>
  20. <action android:name="android.intent.action.MAIN" />
  21. <category android:name="android.intent.category.LAUNCHER" />
  22. </intent-filter>
  23. </activity>
  24. <activity android:name="com.example.photowallfallsdemo.ImageDetailsActivity" >
  25. </activity>
  26. </application>
  27. </manifest>

这样所有的编码工作就已经完成了,现在我们运行一下程序,又会看到熟悉的照片墙界面,点击任意一张图片会进入到相应的大图界面,并且可以通过多点触控的方式对图片进行缩放,放大后还可以通过单指来移动图片,如下图所示。



好了,今天的讲解到此结束,有疑问的朋友请在下面留言。

猜你喜欢

转载自blog.csdn.net/xiaozhude/article/details/80932192