完美解决拍照裁剪图片完成之后崩溃问题

使用拍照,相册选择相片和裁剪功能基本上都是使用系统自带的Intent来实现

步骤:点击按钮,弹出对话框,选择拍照或者从相册选择图片,并且保存到SD卡中,并且显示在Imageview中

首先最终结果存在了imageUri中,但从拍照Intent到裁剪Intent之间结果是怎么传的呢?估计是通过Intent中的data来传的,当数据过大,即超过1M时就崩了!!!!所以我们要想办法分离这个过程,将中间数据先暂存一下放在imageUri  ,然后再调裁剪Intent,最后把结果存在imageCropUri 中。

这样就完美解决拍照裁剪图片完成之后崩溃问题

主要的类MainActivity.java

package com.waka.takephotos;

import android.app.Activity;
import android.app.AlertDialog;
import android.content.DialogInterface;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.Toast;

import java.io.File;

/**
 *
 *  关键:
 *          最终结果存在了imageUri中,但从拍照Intent到裁剪Intent之间结果是怎么传的呢?估计是通过Intent中的data来传的,当数据过大,即超过1M时就崩了!!!!
            所以我们要想办法分离这个过程,将中间数据先暂存一下,然后再调裁剪Intent,最后把结果存在imageCropUri 中。
 *
 *  把拍照的照片,裁剪后的照片保存在sd中,添加sd卡的读写权限
 *
 * 调用拍照应用隐式意图   ---"android.media.action.IMAGE_CAPTURE"
 * 调用相册应用隐式意图   ---"android.intent.action.PICK"
 * 调用裁剪应用隐式意图   ---"com.android.camera.action.CROP"
 */
public class MainActivity extends Activity {

    private Button btn_open_pic;
    private ImageView iv_show_pic;

    // 拍照存放本地文件名称
    public static final String IMAGE_FILE_NAME = "faceImage.jpg";
    // 裁剪后的文件名称
    public static final String IMAGE_FILE_NAME_TEMP = "faceImage_temp.jpg";

    private File file = new File(Environment.getExternalStorageDirectory(), IMAGE_FILE_NAME);
    private Uri imageUri ;

    private File cropFile = new File(Environment.getExternalStorageDirectory(), IMAGE_FILE_NAME_TEMP);
    private Uri imageCropUri = Uri.fromFile(cropFile);
    // 上传头像方式文字数据
    private String[] items = new String[] { "选择本地图片", "拍照" };

    // 请求码
    private static final int IMAGE_REQUEST_CODE = 0;
    private static final int CAMERA_REQUEST_CODE = 2;
    private static final int RESULT_REQUEST_CODE = 3;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        btn_open_pic = (Button) findViewById(R.id.btn_open_pic);
        iv_show_pic = (ImageView) findViewById(R.id.iv_show_pic);
        btn_open_pic.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                showDialog();
            }
        });
    }

    /**
     * 选择是拍照还是选择本地图片
     */
    private void showDialog() {

        new AlertDialog.Builder(this).setTitle("设置头像").setItems(items, new DialogInterface.OnClickListener() {
            @Override
            public void onClick(DialogInterface dialog, int which) {
                switch (which) {
                    case 0:
                        //从相册选择
                        Intent intent = new Intent(Intent.ACTION_PICK, null);
                        intent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
                        startActivityForResult(intent, IMAGE_REQUEST_CODE);
                        break;
                    case 1:
                        //拍照
                        Intent intentFromCapture = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
                        // 判断存储卡是否可以用,可用进行存储
                        if (hasSdcard()) {
                            intentFromCapture.putExtra("return-data", false);
                            imageUri = Uri.fromFile(file);
                            intentFromCapture.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
                            intentFromCapture.putExtra("noFaceDetection", true);
                        }
                        startActivityForResult(intentFromCapture, CAMERA_REQUEST_CODE);
                        break;
                }
            }
        }).setNegativeButton("取消", new DialogInterface.OnClickListener() {

            @Override
            public void onClick(DialogInterface dialog, int which) {
                dialog.dismiss();
            }
        }).show();

    }
    /**
     * 判断sdcard是否存在
     */
    private boolean hasSdcard() {
        // 判断Sdcard是否可用
        if (android.os.Environment.getExternalStorageState().equals(android.os.Environment.MEDIA_MOUNTED)) {
            return true;
        } else
            return false;
    }

    /**
     * 处理回转值
     */
    @SuppressWarnings("static-access")
    @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (resultCode != this.RESULT_CANCELED) {
            switch (requestCode) {
                case IMAGE_REQUEST_CODE:
                    // 裁剪图片方法
                    startPhotoZoom(data.getData());
                    break;
                // 保存头像
                case CAMERA_REQUEST_CODE:
                    if (hasSdcard()) {
                        startPhotoZoom(imageUri);
                    } else {
                        Toast.makeText(this, "未找到存储卡,无法存储照片!", Toast.LENGTH_SHORT).show();
                    }
                    break;
                // 裁剪之后,显示头像
                case RESULT_REQUEST_CODE:
                    if (data != null) {
                        setImageToView(data);
                    }
                    break;
            }
        }
    }

    /**
     *
     * 裁剪图片方法实现
     *
     */
    public void startPhotoZoom(Uri uri) {
        if (uri == null) {
            Log.i("tag", "========The uri is not exist.");
        }
        Intent intent = new Intent("com.android.camera.action.CROP");
        intent.setDataAndType(uri, "image/*");
        // 设置裁剪
        intent.putExtra("crop", "true");
        // aspectX aspectY 是宽高的比例
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        intent.putExtra("scale", true);
        // outputX outputY 是裁剪图片宽高
        intent.putExtra("outputX", 150);
        intent.putExtra("outputY", 150);
        //裁剪之后,保存在裁剪文件中,关键
        intent.putExtra(MediaStore.EXTRA_OUTPUT, imageCropUri);
        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
        intent.putExtra("noFaceDetection", true);
        intent.putExtra("return-data", false);
        startActivityForResult(intent, RESULT_REQUEST_CODE);
    }

    /**
     * 保存裁剪之后的图片数据,并显示
     */
    private void setImageToView(Intent data) {
        Bundle extras = data.getExtras();
        Bitmap bitmap = null;

        try {
            bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageCropUri));
            Log.d("size",bitmap.getByteCount()+"");
            iv_show_pic.setImageBitmap(bitmap);
        } catch (Exception e) {
            e.printStackTrace();
        }
        if (extras != null) {
           // bitmap = extras.getParcelable("data");
           // Log.d("size",bitmap.getByteCount()+"");

			/*
			 * ByteArrayOutputStream stream = new ByteArrayOutputStream();
			 * bitmap.compress(Bitmap.CompressFormat.JPEG, 60, stream);
			 * byte[] b = stream.toByteArray(); // 将图片流以字符串形式存储下来
			 *
			 * tp = new String(Base64Coder.encodeLines(b));
			 * 这个地方大家可以写下给服务器上传图片的实现,直接把tp直接上传就可以了, 服务器处理的方法是服务器那边的事了,吼吼
			 *
			 * 如果下载到的服务器的数据还是以Base64Coder的形式的话,可以用以下方式转换 为我们可以用的图片类型就OK啦...吼吼
			 * Bitmap dBitmap = BitmapFactory.decodeFile(tp);
			 * Drawable drawable = new BitmapDrawable(dBitmap);
			 */
        }
    }


}


布局文件activity_main.xml:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
                xmlns:tools="http://schemas.android.com/tools"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:orientation="vertical"
                tools:context=".MainActivity">
    <Button
        android:id="@+id/btn_open_pic"
        android:layout_gravity="center_horizontal"
        android:text="选择图片"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>
    <ImageView
        android:id="@+id/iv_show_pic"
        android:layout_gravity="center_horizontal"
        android:layout_width="200dp"
        android:layout_height="300dp"/>
</LinearLayout>

添加权限:

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

参考:

http://blog.csdn.net/harvic880925/article/details/43163175

http://blog.csdn.net/ryantang03/article/details/8654137


源码下载地址:

http://download.csdn.net/detail/wk843620202/9522162




  • 3
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值