android 打开相机拍摄照片(包含动态权限、部分手机图片旋转、压缩)

用手机拍摄照片,在android6.0以后需要动态获取权限,部分手机还会出现拍摄完后图片旋转90的问题,比如三星、小米有的机型都会旋转。。。拍摄完的照片还需要压缩后才能上传

步骤为:

1,动态权限获取

2,调起相机拍照

3,获取结果判断是否有图片旋转,若存在旋转则按照相应的角度再恢复正常

4,图片压缩,压缩完成,执行上传

 

1,动态权限获取使用的是Rxpermissions

在app的gradle里:

repositories {
    maven { url "https://jitpack.io" }
}

dependencies {
....
implementation 'com.github.tbruyelle:rxpermissions:0.10.2'
implementation 'io.reactivex.rxjava2:rxandroid:2.1.1'

}

2,正式调相机:

private val mRxPermissions by lazy { RxPermissions(this) }
private val mPhotoUtils by lazy { PhotoUtils(this) }
private var picAuthDir: String? = ""
private var auth_idcard_front_path: String = ""
override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)
    picAuthDir = FileUtils.getImageFileDir(this, "AuthPhoto").toString() + File.separator
    auth_idcard_front_path = picAuthDir + "idCardFront" + System.currentTimeMillis() + ".png"
   
}
mRxPermissions.request(Manifest.permission.CAMERA).subscribe(object : Consumer<Boolean> {
    override fun accept(t: Boolean) {
        if (t) {
            openCamera()
        } else {
            DeviceUtils.openSettingPage(this, 16)
        }
    }

})
/**
 * 开启相机
 */
private fun openCamera() {
    mPhotoUtils.openCamera()
    mPhotoUtils.setPhotoPathListener(object : PhotoUtils.OnPhotoPathListener {
        override fun onGetPath(path: String) {
            //※※※※※此时获取到照片的路径,处理照片
            rotatePic(path)
        }
    })

}
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    super.onActivityResult(requestCode, resultCode, data)
    if (requestCode == PhotoUtils.REQUEST_CAMERA_CODE) {
        mPhotoUtils.onAcitivityResult(requestCode, resultCode, data)
    }
}
拍照工具类:
/**
 * Created by Lyq
 * on 2021-04-14
 */
public class PhotoUtils {

    public static int PHOTO_CAMERA_CODE = 1; //相机
    public static int PHOTO_ALBUM_CODE = 2; //相册

    public static int REQUEST_CAMERA_CODE = 3; //相机requestCode
    public static int REQUEST_ALBUM_CODE = 4; //相册requestCode

    private Activity mActivity;
    private int mSelectType;
    private String mFilePath;//文件存储路径
    private Uri mFileUri;//文件存储uri

    public PhotoUtils(Activity activity) {
        mActivity = activity;
    }

    public void setPhotoType(int type, String filePath) {
        this.mSelectType = type;
        this.mFilePath = filePath;
        File outputImage = new File(filePath);
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
            mFileUri = FileProvider.getUriForFile(mActivity, mActivity.getPackageName() + ".fileProvider", outputImage);
        } else {
            mFileUri = Uri.fromFile(outputImage);
        }
    }



    private OnPhotoPathListener mPhotoPathListener;
    public void setPhotoPathListener(OnPhotoPathListener listener) {
        this.mPhotoPathListener = listener;
    }
    public interface OnPhotoPathListener {
        void onGetPath(String path);
    }


    /**
     * 相机拍照
     */
    public void openCamera() {

        try {
            Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
                intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION);
            }
            intent.putExtra(MediaStore.EXTRA_OUTPUT, mFileUri);

            if (mSelectType == PHOTO_CAMERA_CODE) {
                mActivity.startActivityForResult(intent, REQUEST_CAMERA_CODE);
            }
        } catch (Exception e) {
            e.printStackTrace();
            ToastUtils.showToastShort(ResourceUtils.getString(R.string.camera_error));
            LogUtils.Companion.e(e.toString());
        }

    }

    public void onAcitivityResult(int requestCode,int resultCode,Intent intent){
        if (requestCode == REQUEST_CAMERA_CODE) {
            if(mPhotoPathListener !=null){
                mPhotoPathListener.onGetPath(mFilePath);
            }
        }

    }

}

 

3,获取路径判断是否有图片旋转,矫正

/**
 * 根据旋转角度,把图片旋转回来
 */
private fun rotatePic(path: String) {
    Thread(Runnable {
        //图片 旋转问题
        val bitmapDegree = ImageSelectUtils.getBitmapDegree(path) //小米是90度bug
        val bitmap = BitmapFactory.decodeFile(path)
        if (bitmap == null) {
            return@Runnable
        }

        val bitmapNew: Bitmap = ImageSelectUtils.rotateBitmapByDegree(bitmap, bitmapDegree)

        var filenew: File? = null
        try {
            if (bitmapNew != null && !TextUtils.isEmpty(path)) {
                filenew = ImageSelectUtils.saveAndCoverFile_crm(bitmapNew, path)
            }
        } catch (e: IOException) {
            e.printStackTrace()
        }

        val msg = Message()
        msg.what = 1
        val bundle = Bundle()
        bundle.putSerializable("rotateFile", filenew)
        msg.data = bundle
        handler.sendMessage(msg)
    }).start()
}

 

private val handler = object : Handler() {
    override fun handleMessage(msg: Message) {
        var handlerBundle = Bundle()
        when (msg.what) {
            1 -> {
                handlerBundle = msg.data
                if (handlerBundle != null) {
                    val rotateFile = handlerBundle.getSerializable("rotateFile") as File
                    //※※※※※,根据旋转完的File压缩图片
                    condensePic(this,rotateFile,picAuthDir)
                }
            }

        }
    }
}

处理图片旋转角度,与矫正工具类ImageSelectUtils:

/**
 * Created by Lyq
 * on 2021-04-17
 */
public class ImageSelectUtils {
    /**
     * 读取图片的旋转的角度
     *
     * @param path 图片绝对路径
     * @return 图片的旋转角度
     */
    public static int getBitmapDegree(String path) {
        int degree = 0;
        try {
            // 从指定路径下读取图片,并获取其EXIF信息
            if (!TextUtils.isEmpty(path)) {

                ExifInterface exifInterface = new ExifInterface(path);
                int orientation = exifInterface.getAttributeInt(ExifInterface.TAG_ORIENTATION,
                        ExifInterface.ORIENTATION_NORMAL);
                // 获取图片的旋转信息

                switch (orientation) {
                    case ExifInterface.ORIENTATION_ROTATE_90:
                        degree = 90;
                        break;
                    case ExifInterface.ORIENTATION_ROTATE_180:
                        degree = 180;
                        break;
                    case ExifInterface.ORIENTATION_ROTATE_270:
                        degree = 270;
                        break;
                }
            }

        } catch (IOException e) {
            e.printStackTrace();
        }
        return degree;
    }

    /**
     * 旋转图片,使图片保持正确的方向。
     *
     * @param bitmap  原始图片
     * @param degrees 原始图片的角度
     * @return Bitmap 旋转后的图片
     */
    public static Bitmap rotateBitmapByDegree(Bitmap bitmap, int degrees) {
        if (degrees == 0 || null == bitmap) {
            return bitmap;
        }
        Matrix matrix = new Matrix();
        matrix.setRotate(degrees, bitmap.getWidth() / 2, bitmap.getHeight() / 2);
        Bitmap bmp = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, true);
        bitmap.recycle();
        return bmp;
    }

    public static File saveAndCoverFile_crm(Bitmap bm, String path) throws IOException {//将Bitmap类型的图片转化成file类型
        File dirFile = new File(path);
        if (!dirFile.exists()) {
            dirFile.mkdir();
        }
        File myCaptureFile = new File(path);
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream(myCaptureFile));
        bm.compress(Bitmap.CompressFormat.JPEG, 80, bos);
        bos.flush();
        bos.close();
        return myCaptureFile;

    }

}

4,图片压缩

压缩图片使用的是luban;

implementation 'com.qiniu:qiniu-android-sdk:7.3.15'
/**
 * 压缩
 * @param context 上下文
 * @param file 要压缩的文件
 * @param targetDir 压缩完的目标文件夹
 * @param type
 * @return
 */
public void condensePic(Context context, File file, String targetDir) {
PicCompressUtils.getInstance().with(context).ignoreBy(100).setTargetDir(targetDir).load(file).setCompressListener(new OnCompressListener() {
        @Override
        public void onStart() {

        }

        @Override
        public void onSuccess(File file) {
 //※※※※※ 压缩成功
//压缩成功 进行上传,是传到自己服务器还是三方
        }

        @Override
        public void onError(Throwable e) {
 //※※※※※ 压缩失败
        }

    }).launch();

封装的压缩工具类:

/**
 * Created by Lyq
 * on 2021-04-13
 */
public class PicCompressUtils {
    private static PicCompressUtils picCompressUtils;
    private Luban.Builder builder;

    public static synchronized PicCompressUtils getInstance() {
        if (picCompressUtils == null) {
            picCompressUtils = new PicCompressUtils();
        }
        return picCompressUtils;
    }

    public PicCompressUtils with(Context mContext) {
        if (mContext != null) {
            builder = Luban.with(mContext);
        }
        return this;
    }

    public PicCompressUtils load(String path) {
        if (builder != null && path != null) {
            builder.load(path);
        }
        return this;
    }

    public PicCompressUtils load(File file) {
        if (builder != null && file != null) {
            builder.load(file);
        }
        return this;
    }

    public PicCompressUtils load(List<String> list) {
        if (builder != null && list != null) {
            builder.load(list);
        }
        return this;
    }

    public PicCompressUtils load(InputStreamProvider inputStreamProvider) {
        if (builder != null && inputStreamProvider != null) {
            builder.load(inputStreamProvider);
        }
        return this;
    }


    public PicCompressUtils ignoreBy(int size) {
        if (builder != null) {
            builder.ignoreBy(size);
        }
        return this;
    }

    public PicCompressUtils setTargetDir(String path) {
        if (builder != null && path != null) {
            builder.setTargetDir(path);
        }
        return this;
    }

    public PicCompressUtils filter(CompressionPredicate compressionPredicate) {
        if (builder != null && compressionPredicate != null) {
            builder.filter(compressionPredicate);
        }
        return this;
    }

    public PicCompressUtils setCompressListener(OnCompressListener onCompressListener) {
        if (builder != null && onCompressListener != null) {
            builder.setCompressListener(onCompressListener);
        }
        return this;
    }


    public void launch() {
        if (builder != null) {
            builder.launch();
        }
    }

}

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
Android相机拍摄的图片可能会因为设备方向的不同而导致旋转。这是因为Android相机的预览和拍照方向并不一定相同。解决这个问题的方法如下: 1. 读取图片的 Exif 信息,获取图片的方向信息。Exif 信息是一些存储在图片中的元数据,包括拍摄地点、相机型号、拍摄时间等。其中也包括了图片的方向信息。 2. 根据 Exif 中的方向信息,将图片进行旋转。可以使用 Bitmap 类的 createBitmap() 方法创建一个新的 Bitmap,并将原始图片旋转后绘制到新的 Bitmap 上。 以下是示例代码: ``` public static Bitmap rotateBitmap(Bitmap bitmap, String path) { int degree = getBitmapDegree(path); if (degree != 0) { Matrix matrix = new Matrix(); matrix.postRotate(degree); Bitmap rotatedBitmap = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, true); bitmap.recycle(); return rotatedBitmap; } else { return bitmap; } } public static int getBitmapDegree(String path) { int degree = 0; try { ExifInterface exifInterface = new ExifInterface(path); int orientation = exifInterface.getAttributeInt(ExifInterface.TAG_ORIENTATION, ExifInterface.ORIENTATION_NORMAL); switch (orientation) { case ExifInterface.ORIENTATION_ROTATE_90: degree = 90; break; case ExifInterface.ORIENTATION_ROTATE_180: degree = 180; break; case ExifInterface.ORIENTATION_ROTATE_270: degree = 270; break; } } catch (IOException e) { e.printStackTrace(); } return degree; } ``` 在使用相机拍照后获取图片的时候,可以调用该方法对图片进行旋转,以得到正确的方向。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值