Android实现相机拍摄、选择、图片裁剪功能

最近的一些学习心得:

功能实现:点击圆形头像之后可以实现相册上传或者开启相机,然后把得到的图片经过剪裁,把剪裁过的图片设置为头像的背景图

步骤:第一步:自定义一个类,继承ImageView,重写draw方法,实现外观为圆形

第二步:在xml文件中引用该控件

第三步:实现圆形头像的点击事件,点击后显示对话框界面,询问你是打开相册还是相机(自动省略显示对话框的代码)

第四步:根据用户选择情况,打开相册或者相机

第五步:将拍摄的图片或者相册选中的图片进行剪裁,将结果保存在指定内存区域

第六步:更新头像图片

具体实现:

第一步:自定义一个类,继承ImageView,重写draw方法,实现外观为圆形

//圆形头像类
public class MyRoundPhoto extends ImageView{

 private Paint p;
 private Bitmap bitmap;
 private Context context;
 private int wAndHeight[]=new int[2];
 private File file;

 public MyRoundPhoto(Context context, AttributeSet attrs, int defStyle) {
 super(context, attrs, defStyle);
 // TODO Auto-generated constructor stub

 //获得控件长宽(px)
 wAndHeight = getWidthAndHeight(context,attrs);

 this.context = context;

    //初始化控件
    init();

 }

 public MyRoundPhoto(Context context) {
 super(context);
 // TODO Auto-generated constructor stub

    //获得控件长宽(px)
 wAndHeight=getWidthAndHeight(context,attrs);

 this.context = context;
 init();
 }

 public MyRoundPhoto(Context context, AttributeSet attrs) {
 super(context, attrs);

 //获得控件长宽(px)
 wAndHeight=getWidthAndHeight(context,attrs);

 // TODO Auto-generated constructor stub
 this.context = context;
 init();
 }

 @Override
 protected void onDraw(Canvas canvas) {
 // TODO Auto-generated method stub
 super.onDraw(canvas);
 canvas.drawBitmap(bitmap, new Matrix(), p);

 }

 private void init(){
 //从手机存储区域获取图片文件(该位置为手机相册选中的图片经过剪裁后的图片的存储路径)
 file = new File(Environment.getExternalStorageDirectory(),Info.PHOTO_NAME);

    //如果图片文件存在,则显示,否则则创建并显示
 if(file.exists()){
  Log.v("文件存在", "是");
  this.bitmap = BitmapFactory.decodeFile(file.getAbsolutePath());
 }
 else{
  Log.v("文件不存在", "是");

  //生成默认图片的文件
  this.bitmap=BitmapFactory.decodeStream(context.getResources().openRawResource(R.drawable.defalut));
  //person.setPicture()
  FileOutputStream fos=null;
  try {
  fos = new FileOutputStream(file);
  } catch (FileNotFoundException e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
  }
  bitmap.compress(Bitmap.CompressFormat.PNG, 100, fos); //压缩
  try {
  fos.flush();
  fos.close();
  } catch (IOException e) {
  // TODO Auto-generated catch block
  e.printStackTrace();
  }

 }

 //将方形的位图转换为圆形的位图
 this.bitmap = toRoundBitmap(this.bitmap);
 p = new Paint();

 }

 private Bitmap toRoundBitmap(Bitmap map){

 //int height = map.getHeight()+100;
 int height=convertDIP2PX(context,this.wAndHeight[1]); //位图的高度(px)
 int width = convertDIP2PX(context,this.wAndHeight[0]);//位图的宽度(px)

 //创建画布
 Bitmap bit = Bitmap.createBitmap(width, height, Config.ARGB_8888);
 Canvas canvas = new Canvas(bit);

 //画笔
 Paint paint = new Paint();
 paint.setAntiAlias(false);
 int r = (width>height)?height:width;

 //绘制圆形
 RectF rectF = new RectF(0,0,r,r);
 canvas.drawRoundRect(rectF, r/2, r/2, paint);

 //画头像
 //canvas.drawARGB(0, 0, 0, 0);
 paint.setXfermode(new PorterDuffXfermode(Mode.SRC_IN));
 canvas.drawBitmap(map, null,rectF, paint);

 //返回圆形位图
 return bit;
 }

  //使当前视图无效,从而使系统重新绘制视图
 public void myValidate(){
 bitmap = BitmapFactory.decodeFile(file.getAbsolutePath());
 bitmap=toRoundBitmap(bitmap);
 invalidate();

 }

  //将dp转换为px
 private static int convertDIP2PX(Context context, int dip) {
   float scale = context.getResources().getDisplayMetrics().density;
   return (int)(dip*scale + 0.5f*(dip>=0?1:-1));
 } 

 //根据xml文件中的属性,返回宽高(px)
  private static int[] getWidthAndHeight(Context context,AttributeSet attrs){
 int height,width;
 int n = attrs.getAttributeCount();
 int wAndH[] = new int[2];

 for(int i=0;i<n;i++){
  String str = attrs.getAttributeName(i);

  //获取宽度
  if(str.equals("layout_width")){
  //System.out.println(attrs.getAttributeName(0));
  String sttr = attrs.getAttributeValue(i);
  String temp = "";
  int j=0;
  while(sttr.charAt(j)>='0'&&sttr.charAt(j)<='9'){
   temp+=sttr.charAt(j);
   j++;
  }
  wAndH[0]=Integer.parseInt(temp);
  temp="";
  continue;
  }

  //获取长度
  if(str.equals("layout_height")){
  //System.out.println(attrs.getAttributeName(1));
  String sttr = attrs.getAttributeValue(i);
  String temp = "";
  int j=0;
  while(sttr.charAt(j)>='0'&&sttr.charAt(j)<='9'){
   temp+=sttr.charAt(j);
   j++;
  }
  //System.out.println("temp"+temp);
  wAndH[1]=Integer.parseInt(temp);
  temp="";
  continue;
  }
 }
 return wAndH;
 }
}

第二步:在xml文件中引用该控件

 <com.包名.MyRoundPhoto
      android:id="@+id/myRoundPhoto"
        android:layout_width="100dp"
        android:layout_height="100dp" >

      </com.包名.MyRoundPhoto>

第三步:实现圆形头像的点击事件,点击后显示对话框界面,询问你是打开相册还是相机(自动省略显示对话框的代码)

public void onClick(View v) {
 // TODO Auto-generated method stub

 //点击头像
 if(v.getId()==R.id.myRoundPhoto){
  //打开DialogActivity,询问打开照相机还是相册
  Intent intent = new Intent(GuideActivity.this,DialogActivity.class);
  startActivityForResult(intent, Info.PICK_PHOTO);
 }
 }

 第四步:根据用户选择情况,打开相册或者相机

 image =new File(Environment.getExternalStorageDirectory(),Info.PHOTO_NAME);

  public void onClick(View v) {
 // TODO Auto-generated method stub
 switch(v.getId()){

 //打开相机
 case R.id.imageButton1:{
  Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
  intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(image));
  startActivityForResult(intent, Info.OPEN_CAMERA);
  break;
 }

 //打开相册
 case R.id.imageButton2:{
  Intent intent = new Intent(Intent.ACTION_PICK);
  intent.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(image));
  intent.setType("image/*");
  startActivityForResult(intent, Info.OPEN_GALLERY);
  break;
 }
 }
 }

第五步:将拍摄的图片或者相册选中的图片进行剪裁,将结果保存在指定内存区域

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
 // TODO Auto-generated method stub
 super.onActivityResult(requestCode, resultCode, data);

 switch(requestCode){
 //打开相机
 case Info.OPEN_CAMERA:{
  if(resultCode==RESULT_OK){

  //启动裁剪activity
  Log.v("启动剪裁程序", "是的");
  Intent intent1 = new Intent("com.android.camera.action.CROP");
       intent1.setDataAndType(Uri.fromFile(image), "image/*");
       intent1.putExtra("crop", "true");
       intent1.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(image));//
       intent1.putExtra("aspectX", 1);
       intent1.putExtra("aspectY", 1);
       intent1.putExtra("outputFormat", Bitmap.CompressFormat.JPEG);
       intent1.putExtra("outputX", 720);
       intent1.putExtra("outputY", 720);
       intent1.putExtra("return-data", false);
       startActivityForResult(intent1, Info.CROP_PHOTO);
  }
  break;
 }

 //打开相册
 case Info.OPEN_GALLERY:{

  if(resultCode==RESULT_OK){
  //启动剪裁程序
  Log.v("启动剪裁程序", "是的");
  Intent intent1 = new Intent("com.android.camera.action.CROP");
       intent1.setDataAndType(Uri.fromFile(image), "image/*");
       intent1.putExtra("crop", "true");
       intent1.putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(image));//
       intent1.putExtra("aspectX", 1);
       intent1.putExtra("aspectY", 1);
       intent1.putExtra("outputFormat", Bitmap.CompressFormat.JPEG);
       intent1.putExtra("outputX", 720);
       intent1.putExtra("outputY", 720);
       intent1.putExtra("return-data", false);
       startActivityForResult(intent1, Info.CROP_PHOTO);
  }
  break;
 }

 //裁剪图片
 case Info.CROP_PHOTO:{
  Intent intent=new Intent();
  setResult(this.RESULT_OK, intent);
  finish();
  break;
 }
 }
 }

第六步:更新头像图片

   protected void onActivityResult(int requestCode, int resultCode, Intent data) {
 // TODO Auto-generated method stub
 super.onActivityResult(requestCode, resultCode, data);

 switch(requestCode){
 //选择头像
 case Info.PICK_PHOTO:{
  //如果摄取图片成功
  if(resultCode==RESULT_OK){
  Log.v("requstCodeGuideOne", "PICK_PHOTO");
  btn_choosePhoto.myValidate(); //使原有视图无效,从而使系统重新绘制视图
  }
  break;
 }

 default:{
  break;
 }
 }
 }

注意:需要添加的权限

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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android ImageView实现图片裁剪和显示功能

    首先在layout布局中设置按钮和一个ImageView <Button android:id="@+id/selectimagebtn" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="选择图片" /> <Button android:id="@+id/cutimagebt

  • 基于RxPaparazzo实现图片裁剪、图片旋转、比例放大缩小功能

    前言:基于RxPaparazzo的图片裁剪,图片旋转.比例放大|缩小. 效果: 开发环境:AndroidStudio2.2.1+gradle-2.14.1 涉及知识: 1.Material Design (CardView+CoordinatorLayout+AppBarLayout+NestedScrollView+CollapsingToolbarLayout+Toolbar+FloatingActionButton)使用 2.butterknife注解式开发 3.基于RxJava+RxAn

  • Android开发从相机或相册获取图片裁剪

    废话不多说了,直接给大家贴代码了. package com.only.android.app; import java.io.File; import android.app.Activity; import android.app.AlertDialog; import android.content.DialogInterface; import android.content.Intent; import android.graphics.Bitmap; import android.gr

  • 使用Java代码在Android中实现图片裁剪功能

    前言 Android应用中经常会遇到上传相册图片的需求,这里记录一下如何进行相册图片的选取和裁剪. 相册选取图片 1. 激活相册或是文件管理器,来获取相片,代码如下: private static final int TAKE_PICTURE_FROM_ALBUM = 1; private void takePictureFromAlbum() { Intent intent = new Intent(Intent.ACTION_GET_CONTENT); intent.setType("ima

  • Android图片裁剪功能实现代码

    在Android应用中,图片裁剪也是一个经常用到的功能.Android系统中可以用隐式意图调用系统应用进行裁剪,但是这样做在不同的手机可能表现出不同的效果,甚至在某些奇葩手机上还会出其他更奇怪的问题,所以调用系统功能进行图片裁剪在很多时候对我们来说并不是一个好的选择.这时候就需要我们自己去实现这种裁剪功能了. 功能分析 要完成图片裁剪的功能,我们需要先知道图片裁剪的功能有哪些.图片裁剪之前,我们需要有一个框指示我们需要裁剪的样式合大小.图片显示出来后大小和位置可能并不是我们所期望的,所以我们还需

  • android调用原生图片裁剪后图片尺寸缩放的解决方法

    在安卓开发中,如果对拍照后的图片进行图片裁剪,如果是调用系统的裁剪,如下: /* * 裁剪图片 */ private void cropPhoto() { Intent intent = new Intent("com.android.camera.action.CROP"); Uri uri = Uri.parse("file://" + picSavePath); intent.setDataAndType(uri, "image/*");

  • Android中利用matrix 控制图片的旋转、缩放、移动

    本文主要讲解利用android中Matrix控制图形的旋转缩放移动,具体参见一下代码: 复制代码 代码如下: /**  * 使用矩阵控制图片移动.缩放.旋转  */  public class CommonImgEffectView extends View { private Context context ;      private Bitmap mainBmp , controlBmp ;      private int mainBmpWidth , mainBmpHeight , c

  • Android UI之ImageView实现图片旋转和缩放

    这一篇,给大家介绍一下ImageView控件的使用,ImageView主要是用来显示图片,可以对图片进行放大.缩小.旋转的功能. android:sacleType属性指定ImageVIew控件显示图片的方式,例如:center表示图像以不缩放的方式显示在ImageView控件的中心,如果设置为fitCenter,表示图像按照比例缩放至合适的位置,并在ImageView控件的中心. 首先我们开发一个简单的案例,实现图片的放大缩小和旋转: 先看看实现的效果: 缩放截图1: 缩放截图2: 旋转截图1

  • Android 7.0中拍照和图片裁剪适配的问题详解

    前言 Android 7.0系统发布后,拿到能升级的nexus 6P,就开始了7.0的适配.发现在Android 7.0以上,在相机拍照和图片裁剪上,可能会碰到以下一些错误: Process: com.yuyh.imgsel, PID: 22995 // 错误1 android.os.FileUriExposedException: file:///storage/emulated/0/Android/data/com.yuyh.imgsel/cache/1486438962645.jpg ex

  • Android Tween动画之RotateAnimation实现图片不停旋转效果实例介绍

    主要介绍Android中如何使用rotate实现图片不停旋转的效果.Android 平台提供了两类动画,一类是 Tween 动画,即通过对场景里的对象不断做图像变换(平移.缩放.旋转)产生动画效果:第二类是 Frame 动画,即顺序播放事先做好的图像,跟电影类似.本文分析 Tween动画的rotate实现旋转效果. 在新浪微博客户端中各个操作进行中时activity的右上角都会有个不停旋转的图标,类似刷新的效果,给用户以操作中的提示.这种非模态的提示方式推荐使用,那么下面就分享下如何实现这种效果

随机推荐