美文网首页Android开发
Android调用手机拍照并获得原图

Android调用手机拍照并获得原图

作者: 欢乐的乐 | 来源:发表于2017-03-17 14:40 被阅读316次

因为最近有一个项目需要调用手机拍照功能,然后将图片显示在ImageView中。因为还是小白,所以花了一点时间去研究,在这里总结一下经验。

更新文章:2017/6/9 、因为之前一直都是使用自己的手机去开发(Android 5.1.0),完全没有想到6.0以上的问题,于是借了别人的手机测试了一下,果然代码出问题了,下面的代码已经过Android 7.0真机测试
更新内容:兼容Android 6.0以上的运行时权限系统和Android7.0应用间共享文件行为变更

更新文章:2017/6/22、╮(╯▽╰)╭之前忘了贴兼容的代码上来,万分抱歉。
更新内容:贴上兼容的代码,Android7.0相机开发的注意事项

权限

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

注意:之前权限声明加上了<uses-permission android:name="android.permission.CAMERA"/>
Android7.0调用相机就直接闪退,有点奇怪,也不知道什么原因。既然不加也不影响功能,大家就不要加上CAMERA权限了。

判断是否SD卡

图片拍摄了以后肯定要找个地方保存一下,然后就研究了一会外部存储和内部存储。

//判断是否有SD卡
public boolean existsSDCard(){
    if(Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){
        return true;
    }else{
        return false;
    }
}

如果手机有SD卡则返回true,如果没有则返回false。

获取默认存储位置

String path = Environment.getExternalStorageDirectory().getPath();

一开始百度别人都说这是返回SD卡的默认存储位置,又有人说这是返回内部存储的默认存储位置。太乱了,于是我试验了一下。
过程忘了截图,直接用文字表达。

这个和系统设置中选择的默认存储位置是有关联的

  1. 如果有SD卡,系统默认存储位置也是SD卡,那么获得的位置就是SD卡的位置
  2. 如果有SD卡,系统默认存储位置是内部存储,那么获得的位置就是内部存储的位置
  3. 如果没有SD卡,那就一定是内部存储位置了

热身准备做好了,准备上代码。

调用手机拍照并获得原图

关键代码

private String path;
private String picPath;
private Uri picUri;
private static final int REQUEST_IMAGE = 1;
private static final String[] authCameraArr = {Manifest.permission.WRITE_EXTERNAL_STORAGE};
private static final int authCameraRequestCode = 5;


//判断当前是否具备所需权限
private boolean hasCameraPhoneAuth(){
    PackageManager pm = this.getPackageManager();
    for(String auth: authCameraArr){
        if(pm.checkPermission(auth, this.getPackageName()) != PackageManager.PERMISSION_GRANTED){
            return false;
        }
    }
    return true;
}

//调用系统相机
public void startCamera(){
    //Android 6.0以上需要运行时权限
    if(Build.VERSION.SDK_INT >= 23){
        if(!hasCameraPhoneAuth()){
            this.requestPermissions(authCameraArr, authCameraRequestCode);
            return;
        }
    }
    //获取默认存储地址,而且指定文件夹
    //File.separator 是默认分隔符 \
    path = Environment.getExternalStorageDirectory().getPath();
    picPath = path + File.separator + getPhotoFileName() + ".jpg";
    Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
    //Android7.0文件保存方式改变了
    if(Build.VERSION.SDK_INT < 24){
        picUri = Uri.fromFile(new File(picPath));
        intent.putExtra(MediaStore.EXTRA_OUTPUT, picUri);//将原图的uri传入
        startActivityForResult(intent, Constants.REQUEST_IMAGE);
    }else{
        ContentValues contentValues = new ContentValues(1);
        contentValues.put(MediaStore.Images.Media.DATA, picPath);
        Uri uri = getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, contentValues);
        intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
        startActivityForResult(intent, Constants.REQUEST_IMAGE);
    }
}

//生成照片的名字
private String getPhotoFileName(){
    Date date = new Date(System.currentTimeMillis());
    SimpleDateFormat format = new SimpleDateFormat("yyyyMMdd_HHmmss");
    return "IMG_" + format.format(date);
}


@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if(resultCode == RESULT_OK){
        if(requestCode == Constants.REQUEST_IMAGE){
            try {
                FileInputStream fis = new FileInputStream(picPath); //根据uri的path获得原图的字节流
                Bitmap bitmap = BitmapFactory.decodeStream(fis);//生成原图
                bitmap = this.centerSquareScaleBitmap(bitmap, this.dip2px(100));//将原图裁剪
                //一定要将原图裁剪了才能显示在ImageView
                //至于为什么,我猜测是因为手机拍的照片像素至少有720*1080
                //图片一般是使用ARGB_8888,一个像素点就4个字节,直接显示原图是很费内存的,于是就无法显示了
                iv_pic.setImageBitmap(bitmap);
            }catch(Exception e){
                e.printStackTrace();
            }
        }
    }
}

//运行时权限的回调
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
    super.onRequestPermissionsResult(requestCode, permissions, grantResults);
    if(requestCode == authCameraRequestCode){
        for(int ret: grantResults){
            if(ret == PackageManager.PERMISSION_GRANTED){
                continue;
            }else{
                Toast.makeText(this, "缺少写文件的权限!", Toast.LENGTH_SHORT).show();
                return;
            }
        }
        startCamera();
    }
}

/**
 * 缩放图片裁剪指定大小的正方形
 * @param bitmap 原图
 * @param edgeLength 正方形的边长
 * @return 裁剪后的图片
 */
public Bitmap centerSquareScaleBitmap(Bitmap bitmap, int edgeLength){
    if(null == bitmap || edgeLength <= 0){
        return null;
    }
    Bitmap result = bitmap;
    int width = bitmap.getWidth();
    int height = bitmap.getHeight();
    if(width > edgeLength && height > edgeLength){
        int longerEdge = (int)(edgeLength * Math.max(width, height) / Math.min(width, height));
        int scaleWidth = width > height ? longerEdge : edgeLength;
        int scaleHeight = width > height ? edgeLength : longerEdge;
        Bitmap scaleBitmap;
        try{
            scaleBitmap = Bitmap.createScaledBitmap(bitmap, scaleWidth, scaleHeight, true);
        }catch(Exception e){
            return null;
        }
        //从图片的正中间裁剪
        int xTopLeft = (scaleWidth - edgeLength)/2;
        int yTopLeft = (scaleHeight - edgeLength)/2;
        try{
            result = Bitmap.createBitmap(scaleBitmap, xTopLeft, yTopLeft, edgeLength, edgeLength);
            scaleBitmap.recycle();
        }catch(Exception e){
            return null;
        }
    }
    return result;
}

//将dp转换成px
public int dip2px(int dpSize){
    final float scale = getResources().getDisplayMetrics().density;
    return (int)(dpSize * scale + 0.5f);
}

//生成照片的名字
private String getPhotoFileName(){
    Date date = new Date(System.currentTimeMillis());
    SimpleDateFormat format = new SimpleDateFormat("yyyyMMdd_HHmmss");
    return "IMG_" + format.format(date);
}

好好学习,天天向上。<( ̄oo, ̄)/


Potato_zero.jpg

相关文章

网友评论

    本文标题:Android调用手机拍照并获得原图

    本文链接:https://www.haomeiwen.com/subject/mtuenttx.html