Android实现拍照、选择相册图片并裁剪功能

作者:迟做总比不做强 时间:2022-09-22 12:25:44 

通过拍照或相册中获取图片,并进行裁剪操作,然后把图片显示到ImageView上。
 当然也可以上传到服务器(项目中绝大部分情况是上传到服务器),参考网上资料及结合项目实际情况,
 测试了多款手机暂时没有发现严重问题。代码有注释,直接贴代码:


public class UploadPicActivity extends Activity implements View.OnClickListener {
private Button take_photo_btn;
private Button select_photo_btn;
private ImageView photo_iv;
//使用照相机拍照获取图片
public static final int TAKE_PHOTO_CODE = 1;
//使用相册中的图片
public static final int SELECT_PIC_CODE = 2;
//图片裁剪
private static final int PHOTO_CROP_CODE = 3;
//定义图片的Uri
private Uri photoUri;
//图片文件路径
private String picPath;

@Override
protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_upload_pic);
 initViews();
}

private void initViews() {
 this.take_photo_btn = (Button) findViewById(R.id.take_photo_btn);
 this.take_photo_btn.setOnClickListener(this);
 this.select_photo_btn = (Button) findViewById(R.id.select_photo_btn);
 this.select_photo_btn.setOnClickListener(this);
 this.photo_iv = (ImageView) findViewById(R.id.photo_iv);
}

@Override
public void onClick(View view) {
 switch (view.getId()) {
  //拍照
  case R.id.take_photo_btn:
   picTyTakePhoto();
   break;
  //选择图库
  case R.id.select_photo_btn:
   pickPhoto();
   break;
 }
}

/**
 * 拍照获取图片
 */
private void picTyTakePhoto() {
 //判断SD卡是否存在
 String SDState = Environment.getExternalStorageState();
 if (SDState.equals(Environment.MEDIA_MOUNTED)) {
  Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);//"android.media.action.IMAGE_CAPTURE"
/***
* 使用照相机拍照,拍照后的图片会存放在相册中。使用这种方式好处就是:获取的图片是拍照后的原图,
* 如果不实用ContentValues存放照片路径的话,拍照后获取的图片为缩略图有可能不清晰
*/
  ContentValues values = new ContentValues();
  photoUri = getContentResolver().insert(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
  intent.putExtra(android.provider.MediaStore.EXTRA_OUTPUT, photoUri);
  startActivityForResult(intent, TAKE_PHOTO_CODE);
 } else {
  Toast.makeText(this, "内存卡不存在", Toast.LENGTH_LONG).show();
 }
}

/***
 * 从相册中取图片
 */
private void pickPhoto() {
 Intent intent = new Intent(Intent.ACTION_PICK, null);
 intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
   "image/*");
 startActivityForResult(intent, SELECT_PIC_CODE);
}

@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
 super.onActivityResult(requestCode, resultCode, data);
 if (resultCode == Activity.RESULT_OK) {
  //从相册取图片,有些手机有异常情况,请注意
  if (requestCode == SELECT_PIC_CODE) {
   if (null != data && null != data.getData()) {
    photoUri = data.getData();
    picPath = uriToFilePath(photoUri);
    startPhotoZoom(photoUri, PHOTO_CROP_CODE);
   } else {
    Toast.makeText(this, "图片选择失败", Toast.LENGTH_LONG).show();
   }
  } else if (requestCode == TAKE_PHOTO_CODE) {
   String[] pojo = {MediaStore.Images.Media.DATA};
   Cursor cursor = managedQuery(photoUri, pojo, null, null, null);
   if (cursor != null) {
    int columnIndex = cursor.getColumnIndexOrThrow(pojo[0]);
    cursor.moveToFirst();
    picPath = cursor.getString(columnIndex);
    if (Build.VERSION.SDK_INT < 14) {
     cursor.close();
    }
   }
   if (picPath != null) {
    photoUri = Uri.fromFile(new File(picPath));
    startPhotoZoom(photoUri, PHOTO_CROP_CODE);
   } else {
    Toast.makeText(this, "图片选择失败", Toast.LENGTH_LONG).show();
   }
  } else if (requestCode == PHOTO_CROP_CODE) {
   if (photoUri != null) {
    Bitmap bitmap = BitmapFactory.decodeFile(picPath);
    if (bitmap != null) {
     //这里可以把图片进行上传到服务器操作
     photo_iv.setImageBitmap(bitmap);
    }
   }
  }
 }
}

/**
 * @param
 * @description 裁剪图片
 * @author ldm
 * @time 2016/11/30 15:19
 */
private void startPhotoZoom(Uri uri, int REQUE_CODE_CROP) {
 Intent intent = new Intent("com.android.camera.action.CROP");
 intent.setDataAndType(uri, "image/*");
 // crop=true是设置在开启的Intent中设置显示的VIEW可裁剪
 intent.putExtra("crop", "true");
 // 去黑边
 intent.putExtra("scale", true);
 intent.putExtra("scaleUpIfNeeded", true);
 // aspectX aspectY 是宽高的比例,根据自己情况修改
 intent.putExtra("aspectX", 3);
 intent.putExtra("aspectY", 2);
 // outputX outputY 是裁剪图片宽高像素
 intent.putExtra("outputX", 600);
 intent.putExtra("outputY", 400);
 intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
 //取消人脸识别功能
 intent.putExtra("noFaceDetection", true);
 //设置返回的uri
 intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
 //设置为不返回数据
 intent.putExtra("return-data", false);
 startActivityForResult(intent, REQUE_CODE_CROP);
}

/**
 * @param
 * @description 把Uri转换为文件路径
 * @author ldm
 * @time 2016/11/30 15:22
 */
private String uriToFilePath(Uri uri) {
 //获取图片数据
 String[] proj = {MediaStore.Images.Media.DATA};
 //查询
 Cursor cursor = managedQuery(uri, proj, null, null, null);
 //获得用户选择的图片的索引值
 int image_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
 cursor.moveToFirst();
 //返回图片路径
 return cursor.getString(image_index);
}
}

布局文件长这样:


<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
   android:layout_width="match_parent"
   android:layout_height="match_parent"
   android:orientation="vertical"
   android:padding="16dp">

<Button
 android:id="@+id/take_photo_btn"
 android:layout_width="match_parent"
 android:layout_height="wrap_content"
 android:layout_marginTop="20dp"
 android:gravity="center"
 android:text="拍照"
 android:textSize="16sp"/>

<Button
 android:id="@+id/select_photo_btn"
 android:layout_width="match_parent"
 android:layout_height="wrap_content"
 android:layout_marginTop="20dp"
 android:gravity="center"
 android:text="选择图片"
 android:textSize="16sp"/>

<ImageView
 android:id="@+id/photo_iv"
 android:layout_width="wrap_content"
 android:layout_height="wrap_content"
 android:layout_gravity="center_horizontal"
 android:layout_marginTop="20dp"/>
</LinearLayout>

最后不要忘记在AndroidManifest.xml中添加UploadPicActivity及权限:


<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<uses-permission
android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>
标签:Android,拍照,选择图片,裁剪图片
0
投稿

猜你喜欢

  • Java的JSTL标签库详解

    2023-07-13 21:58:35
  • Java线程同步方法实例总结

    2022-08-20 20:35:08
  • string与stringbuilder两者的区别

    2021-11-26 00:01:06
  • Java实现答答租车系统

    2022-07-12 01:38:17
  • 详解JAVA 弱引用

    2022-03-12 01:30:29
  • Android仿水波纹流量球进度条控制器

    2022-09-02 01:13:03
  • C#静态构造函数用法实例分析

    2022-03-18 10:32:34
  • Java算法实战之排一亿个随机数

    2022-11-22 02:47:26
  • java8 多个list对象用lambda求差集操作

    2022-02-19 06:26:52
  • springboot启动脚本start.sh和停止脚本 stop.sh的详细教程

    2022-10-11 08:28:26
  • 谈谈变量命名规范的重要性

    2021-08-10 22:40:13
  • Java Annotation注解相关原理代码总结

    2023-11-18 02:13:57
  • Java中DataInputStream和DataOutputStream的使用方法

    2023-02-03 04:08:28
  • java中的FileInputStream三种read()函数用法

    2021-10-27 23:01:21
  • seata的部署和集成详细介绍

    2023-06-05 08:33:24
  • C#集合本质之队列的用法详解

    2023-03-17 06:42:38
  • C#实现跑马灯效果的示例代码

    2022-06-10 07:12:40
  • C#反射应用实例

    2023-11-03 14:47:46
  • java解析多层嵌套json字符串问题

    2023-08-26 11:08:33
  • Spring之IOC详解

    2022-10-13 19:10:24
  • asp之家 软件编程 m.aspxhome.com