接到一个需求,需要做一个类似二维码扫一扫功能的需求,需要将屏幕中的特定区域截图发送到服务器。话不多说先上效果图:
实现思路:获取扫描框的位置,然后在图片上面裁剪。然而就是这么一个简单的思路在适配上面问题多了。首先是surfaceView预览在部分手机上面会出现变形,其次,得到了框的起始点和大小还是裁剪不出特定区域的图片。如果变形怎么裁剪发送到服务端那边的图片还是不正确。所以首先需要解决的便是:surfaceView预览变形的问题:
之前用的求解最佳预览大小的算法不行。网上很多人推荐这个算法,亲测不行!算法如下:
public Size getPropPreviewSize(List 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 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了。