Android--利用相机或相册截取用户头像(解决了miui无法截取,以及部分机型拍照无返回Uri)【上】

声明:

本文的Demo可用于从本地获取用户头像时使用,解决了有些手机系统相机拍照后获取不到拍摄照片的问题,以及解决小米miui系统调用系统裁剪图片功能camera.action.CROP后崩溃或重新打开app的问题。

主活动文件ChooseImageMainActivity

package com.example.no_clay.demolist.ChooseImage;

import android.content.Context;
import android.content.Intent;
import android.content.SharedPreferences;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.drawable.BitmapDrawable;
import android.graphics.drawable.Drawable;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.MediaStore;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.Gravity;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.Toast;

import com.example.no_clay.demolist.R;

import java.io.File;
import java.io.IOException;

import cn.bmob.v3.Bmob;
import cn.bmob.v3.datatype.BmobFile;
import cn.bmob.v3.listener.SaveListener;
import cn.bmob.v3.listener.UploadFileListener;

/**
 * Created by 寒 on 2016/6/4.
 */
public class ChooseImageMainActivity extends AppCompatActivity {
    private ImageView imageView;
    private SelectPicPopupWindow menuWindow;
    private static final int REQUEST_CODE_PICK_IMAGE = 0;
    private static final int REQUEST_CODE_CAPTURE_CAMEIA = 1;
    private static final int RESIZE_REQUEST_CODE = 2;
    private static final String TAG = "ChooseImageMainActivity";
    private Uri userImageUri;//保存用户头像的uri
    private Context context = ChooseImageMainActivity.this;

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.choose_image_main_layout);


        imageView = (ImageView) findViewById(R.id.choose_image_image);
        imageView.setOnClickListener(new View.OnClickListener() {//给ImageView设置点击监听
            @Override
            public void onClick(View v) {
                menuWindow = new SelectPicPopupWindow(context, new
                        View.OnClickListener() {
                            @Override
                            public void onClick(View v) {
                                menuWindow.dismiss();
                                switch (v.getId()){
                                    case R.id.takePhotoBtn: {
                                        String state = Environment.getExternalStorageState();

                                        if (state.equals(Environment.MEDIA_MOUNTED)) {
                                            Intent getImageByCamera = new
                                                    Intent("android.media.action.IMAGE_CAPTURE");
                                            startActivityForResult(getImageByCamera,
                                                    REQUEST_CODE_CAPTURE_CAMEIA);
                                        }
                                        else {
                                            Toast.makeText(getApplicationContext(),
                                                    "请确认已经插入SD卡", Toast.LENGTH_LONG).show();
                                        }
                                        break;
                                    }
                                    case R.id.pickPhotoBtn:
//                                Intent intent = new Intent(Intent.ACTION_PICK);//从相册中选取图片
                                        Intent intent = new Intent("android.intent.action.GET_CONTENT");//从相册/文件管理中选取图片
                                        intent.setType("image/*");//相片类型
                                        startActivityForResult(intent, REQUEST_CODE_PICK_IMAGE);
                                        break;
                                    case R.id.cancelBtn:{
                                        break;
                                    }
                                }
                            }
                        });
                menuWindow.showAtLocation(findViewById(R.id.mainLayout),
                        Gravity.BOTTOM | Gravity.CENTER_HORIZONTAL, 0, 0);
            }
        });
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        Uri imageUri = null;
        if(resultCode == RESULT_CANCELED){
            Toast.makeText(context,"获取失败",Toast.LENGTH_SHORT).show();
            imageView.setImageBitmap(null);//预设的图片
        }else if(resultCode == RESULT_OK) {//选取成功后进行裁剪
            if (requestCode == REQUEST_CODE_PICK_IMAGE) {
                //从图库中选择图片作为头像
                imageUri = data.getData();
                Log.d(TAG, "onActivityResult: " + imageUri);
                reSizeImage(imageUri);
            } else if (requestCode == REQUEST_CODE_CAPTURE_CAMEIA) {
                //使用相机获取头像
                Log.d(TAG, "onActivityResult: from photo");
                imageUri = data.getData();
                Log.d(TAG, "onActivityResult: " + imageUri);
                if (imageUri == null) {
                    //use bundle to get data
                    Bundle bundle = data.getExtras();
                    if (bundle != null) {
                        Bitmap bitMap = (Bitmap) bundle.get("data"); //get bitmap
                        imageUri = Uri.parse(MediaStore.Images.Media.
                                insertImage(getContentResolver(), bitMap, null,null));
                        Log.d(TAG, "onActivityResult: bndle != null" + imageUri);
                        reSizeImage(imageUri);
                    } else {
                        Toast.makeText(getApplicationContext(), "Error", Toast.LENGTH_SHORT).show();
                    }
                }
            }else if(requestCode == RESIZE_REQUEST_CODE){
                Log.d(TAG, "onActivityResult: " + userImageUri);
                showImage(userImageUri);
            }
        }
    }

    private void reSizeImage(Uri uri) {//重新剪裁图片的大小
        Intent intent = new Intent("com.android.camera.action.CROP");
        intent.setDataAndType(uri, "image/*");
        intent.putExtra("crop", "true");//可以裁剪
        intent.putExtra("aspectX", 1);//宽高比例
        intent.putExtra("aspectY", 1);
        intent.putExtra("outputX", 100);
        intent.putExtra("outputY", 100);
        /**
         * 此方法返回的图片只能是小图片(测试为高宽160px的图片)
         * 故将图片保存在Uri中,调用时将Uri转换为Bitmap,此方法还可解决miui系统不能return data的问题
         */
//        intent.putExtra("return-data", true);
//        intent.putExtra("output", Uri.fromFile(new File("/mnt/sdcard/temp")));//保存路径
        userImageUri = Uri.parse("file://"+ Environment.getExternalStorageDirectory().getPath() + "/" + "small.jpg");
        intent.putExtra(MediaStore.EXTRA_OUTPUT, userImageUri);
        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
        startActivityForResult(intent, RESIZE_REQUEST_CODE);
    }

    private void showImage(Uri uri) {
        Log.d(TAG, "showImage: ");
        try {
            Log.d(TAG, "showImage: " + uri.toString());
            SharedPreferences.Editor editor = getSharedPreferences("image",MODE_PRIVATE).edit();
            editor.putString("imageUri",uri.toString());
            editor.commit();//保存头像的uri
            Bitmap photo = MediaStore.Images.Media.getBitmap(context
                    .getContentResolver(), uri);
            imageView.setImageDrawable(new BitmapDrawable(photo));
            Toast.makeText(context,"头像设置成功",Toast.LENGTH_SHORT).show();
        } catch (IOException e) {
            Toast.makeText(context,"头像设置失败",Toast.LENGTH_SHORT).show();
            e.printStackTrace();
        }
    }
}

【注解】:

1.Environment.getExternalStorageState是用来获取外部存储设备的状态,即SD卡的状态,这里为大家推荐一篇博文:[Android中的Environment.getExternalStorageState使用]
(http://blog.csdn.net/yuzhiboyi/article/details/8645730)
2.android.media.action.IMAGE_CAPTURE为调用系统照相机
3.Itent.ACTION_PICK 、Intent.ACTION_GET_CONTENT

两者都可以弹出一张选择列表, 两者都可以当作我们获取手机本地资源的途径,通过设置Intent.setType(String type);即可实现不同资源的选取,不同的是Intent.ACTION_GET_CONTENT会弹出对话框选择资源的来源。常用的类型有:

   //选择图片 requestCode 返回的标识
  Intent innerIntent = new Intent(Intent.ACTION_GET_CONTENT); //"Android.intent.action.GET_CONTENT"
  innerIntent.setType(contentType); //查看类型 String IMAGE_UNSPECIFIED = "image/*";
  Intent wrapperIntent = Intent.createChooser(innerIntent, null);
  ((Activity) context).startActivityForResult(wrapperIntent, requestCode);



  //视频
  Intent innerIntent = new Intent(Intent.ACTION_GET_CONTENT);
  innerIntent.setType(contentType); //String VIDEO_UNSPECIFIED = "video/*";
  Intent wrapperIntent = Intent.createChooser(innerIntent, null);
  ((Activity) context).startActivityForResult(wrapperIntent, requestCode);



  //添加音频
  Intent innerIntent = new Intent(Intent.ACTION_GET_CONTENT);
  innerIntent.setType(contentType); //String VIDEO_UNSPECIFIED = "video/*";
  Intent wrapperIntent = Intent.createChooser(innerIntent, null);
  ((Activity) context).startActivityForResult(wrapperIntent, requestCode);



  //录音
  Intent intent = new Intent(Intent.ACTION_GET_CONTENT);
  intent.setType(ContentType.AUDIO_AMR); //String AUDIO_AMR = "audio/amr";
  intent.setClassName("com.android.soundrecorder",
  "com.android.soundrecorder.SoundRecorder");
  ((Activity) context).startActivityForResult(intent, requestCode);

更详细的使用请看博客:android之Itent.ACTION_PICK Intent.ACTION_GET_CONTENT妙用

4.menuWindow.dismiss();//在点击空白处的时候弹出窗口消失
5.在onActivityResult(int requestCode, int resultCode, Intent data)中为什么要写出两种方法来获取相机拍照后的Uri?

前边通过图片选择返回一个Uri,但是部分机型在拍照在Intent中返回的为Uri为空,于是我们需要将这些机型的获取Uri方法写出来,通过获取存储在bundle中的缩略图的Uri实现。(有些机型利用Bundle来返回照片数据)

6.reSizeImage(Uri uri)中为何不使用下列方法:
Intent intent = new Intent("com.android.camera.action.CROP");
intent.setDataAndType(uri, "image/*");
intent.putExtra("crop", "true");//可以裁剪
intent.putExtra("aspectX", 1);//宽高比例
intent.putExtra("aspectY", 1);
intent.putExtra("outputX", 100);
intent.putExtra("outputY", 100);
intent.putExtra("return-data", true);
intent.putExtra("output", Uri.fromFile(new File("/mnt/sdcard/temp")));//保存路径
startActivityForResult(intent, RESIZE_REQUEST_CODE);

上述方法中,裁剪后的图片通过Intent的putExtra(“return-data”,true)方法进行传递,miui系统问题就出在这里,return-data的方式只适用于小图,miui系统默认的裁剪图片可能裁剪得过大,或对return-data分配的资源不足,造成return-data失败。
解决思路是:裁剪后,将裁剪的图片保存在Uri中,在onActivityResult()方法中,再提取对应的Uri图片转换为Bitmap使用。
其实大家直观也能感觉出来,Intent主要用于不同Activity之间通信,是一种动态的小巧的资源占用,类似于Http请求中的GET,并不适用于传递图片之类的大数据。于是当A生成一个大数据要传递给B,往往不是通过Intent直接传递,而是在A生成数据的时候将数据保存到C,B再去调用C,C相当于一个转换的中间件。

即以下方法:

        Intent intent = new Intent("com.android.camera.action.CROP");
        intent.setDataAndType(uri, "image/*");
        intent.putExtra("crop", "true");//可以裁剪
        intent.putExtra("aspectX", 1);//宽高比例
        intent.putExtra("aspectY", 1);
        intent.putExtra("outputX", 100);
        intent.putExtra("outputY", 100);
        userImageUri = Uri.parse("file://"+ Environment.getExternalStorageDirectory().getPath() + "/" + "small.jpg");
        intent.putExtra(MediaStore.EXTRA_OUTPUT, userImageUri);
        intent.putExtra("outputFormat", Bitmap.CompressFormat.JPEG.toString());
        startActivityForResult(intent, RESIZE_REQUEST_CODE);

主界面布局代码

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical"
    android:id="@+id/mainLayout"
    android:backgroundTint="@color/bisque"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <TextView
        android:textColor="@color/white"
        android:paddingLeft="20dp"
        android:gravity="center_vertical|left"
        android:background="@color/brown"
        android:textSize="20sp"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:text="选择图片" />

    <ImageView
        android:id="@+id/choose_image_image"
        android:layout_marginTop="50dp"
        android:layout_gravity="center_horizontal"
        android:layout_width="100dp"
        android:layout_height="100dp" />

</LinearLayout>

弹出窗口代码:

package com.example.no_clay.demolist.ChooseImage;

import android.annotation.SuppressLint;
import android.content.Context;
import android.graphics.drawable.ColorDrawable;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Button;
import android.widget.PopupWindow;

import com.example.no_clay.demolist.R;


/**
 * Created by 寒 on 2016/6/4.
 */
class SelectPicPopupWindow extends PopupWindow {
    private Button takePhotoBtn, pickPhotoBtn, cancelBtn;
    private View mMenuView;

    @SuppressLint("InflateParams")
    public SelectPicPopupWindow(Context context, View.OnClickListener itemsOnClick) {
        super(context);
        LayoutInflater inflater = (LayoutInflater) context
                .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        mMenuView = inflater.inflate(R.layout.choose_image_layout_dialog_pic, null);
        takePhotoBtn = (Button) mMenuView.findViewById(R.id.takePhotoBtn);
        pickPhotoBtn = (Button) mMenuView.findViewById(R.id.pickPhotoBtn);
        cancelBtn = (Button) mMenuView.findViewById(R.id.cancelBtn);

        cancelBtn.setOnClickListener(itemsOnClick);
        pickPhotoBtn.setOnClickListener(itemsOnClick);
        takePhotoBtn.setOnClickListener(itemsOnClick);


        this.setContentView(mMenuView);

        this.setWidth(ViewGroup.LayoutParams.MATCH_PARENT);

        this.setHeight(ViewGroup.LayoutParams.WRAP_CONTENT);

        this.setFocusable(true);

        this.setAnimationStyle(R.style.PopupAnimation);
        ColorDrawable dw = new ColorDrawable(0x80000000);

        this.setBackgroundDrawable(dw);

        mMenuView.setOnTouchListener(new View.OnTouchListener() {

            @Override
            @SuppressLint("ClickableViewAccessibility")
            public boolean onTouch(View v, MotionEvent event) {

                int height = mMenuView.findViewById(R.id.pop_layout).getTop();
                int y = (int) event.getY();
                if (event.getAction() == MotionEvent.ACTION_UP) {
                    if (y < height) {
                        dismiss();
                    }
                }
                return true;
            }
        });
    }
}

弹出窗口布局choose_image_layout_dialog_pic.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:gravity="center_horizontal"
    android:orientation="vertical" >

    <LinearLayout
        android:id="@+id/pop_layout"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"
        android:background="#312E3F"
        android:gravity="center_horizontal"
        android:orientation="vertical" >

        <Button
            android:id="@+id/takePhotoBtn"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginLeft="10dip"
            android:layout_marginRight="10dip"
            android:layout_marginTop="10dp"
            android:background="#373447"
            android:padding="10dp"
            android:text="拍照"
            android:textColor="#CEC9E7"
            android:textSize="18sp"
            android:textStyle="bold" />

        <Button
            android:id="@+id/pickPhotoBtn"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginLeft="10dip"
            android:layout_marginRight="10dip"
            android:layout_marginTop="5dp"
            android:background="#373447"
            android:padding="10dp"
            android:text="从相册选择"
            android:textColor="#CEC9E7"
            android:textSize="18sp"
            android:textStyle="bold" />

        <Button
            android:id="@+id/cancelBtn"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginBottom="15dip"
            android:layout_marginLeft="10dip"
            android:layout_marginRight="10dip"
            android:layout_marginTop="20dp"
            android:background="@android:color/white"
            android:padding="10dp"
            android:text="取消"
            android:textColor="#373447"
            android:textSize="18sp"
            android:textStyle="bold" />
    </LinearLayout>

</RelativeLayout>
  • 1
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是一个示例代码,用于在Android Kotlin中更换用户头像并从相册选择或拍摄照片: ``` //在此处声明变量 private lateinit var profileImage: ImageView //在onCreate方法中初始化变量 override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_main) profileImage = findViewById(R.id.profile_image) //添加一个点击事件监听器,当用户点击头像时,将弹出一个对话框,询问用户是否从相机相册中选择照片 profileImage.setOnClickListener { val options = arrayOf<CharSequence>("拍照", "从相册选择") val builder = AlertDialog.Builder(this) builder.setTitle("选择图片来源") builder.setItems(options) { dialog, item -> when { options[item] == "拍照" -> { val takePictureIntent = Intent(MediaStore.ACTION_IMAGE_CAPTURE) startActivityForResult(takePictureIntent, 1) } options[item] == "从相册选择" -> { val pickPhotoIntent = Intent(Intent.ACTION_PICK, MediaStore.Images.Media.EXTERNAL_CONTENT_URI) startActivityForResult(pickPhotoIntent, 2) } } } builder.show() } } //在activity中添加以下方法,用于处理从相机相册返回的结果 override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) { super.onActivityResult(requestCode, resultCode, data) if (resultCode == Activity.RESULT_OK) { when (requestCode) { 1 -> { val imageBitmap = data?.extras?.get("data") as Bitmap profileImage.setImageBitmap(imageBitmap) } 2 -> { val selectedImage = data?.data profileImage.setImageURI(selectedImage) } } } } ``` 该代码将在用户点击头像时,弹出一个对话框,询问用户是否从相机相册中选择照片。如果用户选择拍照,将启动相机应用程序并允许用户拍摄照片。如果用户选择从相册选择,将启动相册应用程序并允许用户选择照片。一旦用户选择了照片,将在ImageView中显示所选照片。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值