接到一个需求,需要做一个类似二维码扫一扫功能的需求,需要将屏幕中的特定区域截图发送到服务器。话不多说先上效果图:
实现思路:获取扫描框的位置,然后在图片上面裁剪。然而就是这么一个简单的思路在适配上面问题多了。首先是surfaceView预览在部分手机上面会出现变形,其次,得到了框的起始点和大小还是裁剪不出特定区域的图片。如果变形怎么裁剪发送到服务端那边的图片还是不正确。所以首先需要解决的便是:surfaceView预览变形的问题:
之前用的求解最佳预览大小的算法不行。网上很多人推荐这个算法,亲测不行!算法如下:
public Size getPropPreviewSize(List<Size> list, float th, int minHeight){ Collections.sort(list, sizeComparator); int i = 0; for(Size s:list){ if((s.height >= minHeight) && equalRate(s, th)){ Log.i(TAG, "PreviewSize:w = " + s.width + "h = " + s.height); break; } i++; } if(i == list.size()){ i = 0;//如果没找到,就选最小的size } return list.get(i); }可行的算法需要将预览大小与手机的分辨率挂钩,才能够在不同分辨率的手机上面预览都不变形。算法如下:
/** * 获取最佳预览大小 * @param parameters 相机参数 * @param screenResolution 屏幕宽高 * @return */ private Point getBestCameraResolution(Camera.Parameters parameters, Point screenResolution) { float tmp = 0f; float mindiff = 100f; float x_d_y = (float) screenResolution.x / (float) screenResolution.y; Size best = null; List<Size> supportedPreviewSizes = parameters.getSupportedPreviewSizes(); for (Size s : supportedPreviewSizes) { tmp = Math.abs(((float) s.height / (float) s.width) - x_d_y); if (tmp < mindiff) { mindiff = tmp; best = s; } } return new Point(best.width, best.height); }
获取屏幕宽高的方法如下:
/** * 获取屏幕宽度和高度,单位为px * @param context * @return */ public static Point getScreenMetrics(Context context){ DisplayMetrics dm =context.getResources().getDisplayMetrics(); int w_screen = dm.widthPixels; int h_screen = dm.heightPixels; return new Point(w_screen, h_screen); }调用方法如下:
mParams.setPreviewSize(screenMetrics.x, screenMetrics.y);如此这般,不同手机的预览就不会变形了。
现在要解决的便是裁剪了。
如果只是获取框的大小和起始点,那么不同分辨率的手机还是会出现裁剪误差的情况,那么如何有效的解决呐?这个时候就需要将图片的成像大小和屏幕宽高的比例算出来,然后将起始点和大小都对应算出来,因为我们算出来的起始点或者是大小都只是在屏幕上面的,而其成像的大小和分辨率的大小有可能不是一致,会有一个比例在。刚开始用红米note2测试,都是可以裁剪上去,没想到换了一个三星的直接懵逼了,裁剪歪了。后来打印出来发现红米note2的成像大小和分辨率大小刚好是1,所以裁剪的时候正好刚刚好。而三星的比例不是1,裁剪就出了问题。贴出关键的算法:
/** * 将屏幕中的位置(大小)对应到图片中 * * @param w 屏幕中的位置(宽度) * @param h 屏幕中的位置(高度) * @return */ private Point createCenterPictureRect(int w, int h) { int wScreen = DisplayUtil.getScreenMetrics(mContext).x; int hScreen = DisplayUtil.getScreenMetrics(mContext).y; int wSavePicture = mCameraInterface.doGetPrictureSize().y; //因为图片旋转了,所以此处宽高换位 int hSavePicture = mCameraInterface.doGetPrictureSize().x; //因为图片旋转了,所以此处宽高换位 float wRate = (float) (wSavePicture) / (float) (wScreen); float hRate = (float) (hSavePicture) / (float) (hScreen); float rate = (wRate <= hRate) ? wRate : hRate;//也可以按照最小比率计算 int wRectPicture = (int) (w * wRate); int hRectPicture = (int) (h * hRate); return new Point(wRectPicture, hRectPicture); }调用到的获取成像大小的方法如下:
public Point doGetPrictureSize() { Size s = mCamera.getParameters().getPictureSize(); return new Point(s.width, s.height); }剩下的就是把拍照成功之后Bitmap进行裁剪就好了。
Bitmap rectBitmap = Bitmap.createBitmap(rotaBitmap, pointX, pointY, pictureWidth, pictureHeight);
在界面中获取扫描框的起始点,大小,传入就OK了。