接到一个需求,需要做一个类似二维码扫一扫功能的需求,需要将屏幕中的特定区域截图发送到服务器。话不多说先上效果图:

Android 代码 屏幕截屏 android 截取部分图片_屏幕宽高

实现思路:获取扫描框的位置,然后在图片上面裁剪。然而就是这么一个简单的思路在适配上面问题多了。首先是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了。