Android 拍照或从相册取图片并裁剪

          android拍照或从相册取图片并裁剪,这个功能很常用,网上也有不少例子,可是很多之前的代码已经都不好用了。这个可能与4.4之后,相册返回的地址和原来不一样有关系吧,下面是我的代码,在nexus6中测试通过。

   

  activity_album.xml布局文件  

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    app:layout_behavior="@string/appbar_scrolling_view_behavior"
    tools:context="com.example.cg.cameralearn.AlbumActivity"
    tools:showIn="@layout/activity_album"
    android:orientation="vertical">

    <ImageView
        android:id="@+id/img_album"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"/>

</LinearLayout>

  menu_album.xml菜单布局

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto">
    <item
        android:id="@+id/menu_openCameral"
        android:title="打开相机"
        app:showAsAction="never" />
    <item
        android:id="@+id/menu_openAlbum"
        android:title="打开相册"
        app:showAsAction="never" />
</menu>

    AlbumActivity.java

package com.example.cg.cameralearn;

import android.content.Context;
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.support.design.widget.FloatingActionButton;
import android.support.design.widget.Snackbar;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.util.Log;
import android.view.Menu;
import android.view.MenuItem;
import android.view.View;
import android.widget.ImageView;

import java.io.File;

public class AlbumActivity extends AppCompatActivity {

    public static void AlbumActivityStart(Context context)
    {
        Intent intent = new Intent();
        intent.setClass(context, AlbumActivity.class);
        context.startActivity(intent);
    }

    private ImageView img_album;
    // 拍照成功,读取相册成功,裁减成功
    private final int  ALBUM_OK = 1, CAMERA_OK = 2,CUT_OK = 3;
    private File file;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_album);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
        fab.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG)
                        .setAction("Action", null).show();
            }
        });

        // 定义拍照后存放图片的文件位置和名称,使用完毕后可以方便删除
        file = new File(Environment.getExternalStorageDirectory(), "temp.jpg");
        file.delete();// 清空之前的文件

        initControls();
    }

    /**
     * 初始化控件
     */
    private void initControls() {
        img_album = (ImageView)findViewById(R.id.img_album);
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {

        getMenuInflater().inflate(R.menu.menu_album, menu);
        return true;
    }

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {

        int id = item.getItemId();

        switch (id)
        {
            case R.id.menu_openCameral:

                //这里被注掉的,是在6.0中进行权限判断的,大家可以根据情况,自行加上
                /*if (ContextCompat.checkSelfPermission(this, Manifest.permission.WRITE_EXTERNAL_STORAGE)
                        != PackageManager.PERMISSION_GRANTED) {
                    //申请WRITE_EXTERNAL_STORAGE权限
                    ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},
                            123);
                    Log.e("Album","我没有权限啊");
                }else {

                    Log.e("Album","我有权限啊");
                }*/

                // 来自相机
                Intent cameraIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
                // 下面这句指定调用相机拍照后的照片存储的路径
                cameraIntent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file));
                startActivityForResult(cameraIntent, CAMERA_OK);// CAMERA_OK是用作判断返回结果的标识
                break;
            case R.id.menu_openAlbum:
                // 来自相册
                Intent albumIntent = new Intent(Intent.ACTION_PICK, null);
                /**
                 * 下面这句话,与其它方式写是一样的效果,如果:
                 * intent.setData(MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
                 * intent.setType(""image/*");设置数据类型
                 * 要限制上传到服务器的图片类型时可以直接写如:"image/jpeg 、 image/png等的类型"
                 */
                albumIntent.setDataAndType(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, "image/*");
                startActivityForResult(albumIntent, ALBUM_OK);
                break;
        }

        return super.onOptionsItemSelected(item);
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        System.out.println("requestCode = " + requestCode);
        switch (requestCode) {
            // 如果是直接从相册获取
            case ALBUM_OK:
                //从相册中获取到图片了,才执行裁剪动作
                if (data != null) {
                    clipPhoto(data.getData(),ALBUM_OK);
                    //setPicToView(data);
                }
                break;
            // 如果是调用相机拍照时
            case CAMERA_OK:
                // 当拍照到照片时进行裁减,否则不执行操作
                if (file.exists()) {
                    clipPhoto(Uri.fromFile(file),CAMERA_OK);//开始裁减图片
                }
                break;
            // 取得裁剪后的图片,这里将其设置到imageview中
            case CUT_OK:
                /**
                 * 非空判断大家一定要验证,如果不验证的话, 在剪裁之后如果发现不满意,
                 * 要重新裁剪,丢弃 当前功能时,会报NullException
                 */

                if (data != null) {
                    setPicToView(data);
                }else
                {
                    Log.e("Main","data为空");
                }
                break;
            default:
                break;

        }
        super.onActivityResult(requestCode, resultCode, data);
    }

    /**
     * 裁剪图片方法实现
     * @param uri          图片uri
     * @param type         类别:相机,相册
     */
    public void clipPhoto(Uri uri,int type) {


        Intent intent = new Intent("com.android.camera.action.CROP");
        intent.setDataAndType(uri, "image/*");
        // 下面这个crop = true是设置在开启的Intent中设置显示的VIEW可裁剪
        intent.putExtra("crop", "true");
        // aspectX aspectY 是宽高的比例,这里设置的是正方形(长宽比为1:1)
        intent.putExtra("aspectX", 1);
        intent.putExtra("aspectY", 1);
        // outputX outputY 是裁剪图片宽高
        intent.putExtra("outputX", 200);
        intent.putExtra("outputY", 200);
        intent.putExtra("return-data", false);

        /**
         * 此处做一个判断
         * 1,相机取到的照片,我们把它做放到了定义的目录下。就是file
         * 2,相册取到的照片,这里注意了,因为相册照片本身有一个位置,我们进行了裁剪后,要给一个裁剪后的位置,
         *   不然onActivityResult方法中,data一直是null
         */
        if(type==CAMERA_OK)
        {
            intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);
        }else {
            intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(file));
        }
        startActivityForResult(intent, CUT_OK);
    }

    /**
     * 保存裁剪之后的图片数据 将图片设置到imageview中
     *
     * @param picdata          资源
     */
    private void setPicToView(Intent picdata) {

        try
        {
            Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(picdata.getData()));
            img_album.setImageBitmap(bitmap);
        }catch (Exception ex)
        {
            ex.printStackTrace();
        }
    }

}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值