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

 更新时间:2021年09月18日 10:35:56   作者:FlyinTang  
自定义控件,重写ImageView 功能实现:点击圆形头像之后可以实现相册上传或者开启相机,然后把得到的图片经过剪裁,把剪裁过的图片设置为头像的背景图,需要的朋友可以参考下

最近的一些学习心得:

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

步骤:第一步:自定义一个类,继承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"/>

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

相关文章

  • Android开发之TableLayout表格布局

    Android开发之TableLayout表格布局

    这篇文章主要为大家详细介绍了Android开发之TableLayout表格布局,表格布局模型是以行列的形式管理子控件,对TableLayout表格布局感兴趣的小伙伴们可以参考一下
    2016-03-03
  • Android和iOS包批量重签名

    Android和iOS包批量重签名

    这篇文章主要为大家详细介绍了Android和iOS包批量重签名,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-07-07
  • 详解Android6.0运行时权限管理

    详解Android6.0运行时权限管理

    自从Android6.0发布以来,在权限上做出了很大的变动,不再是之前的只要在manifest设置就可以任意获取权限,而是更加的注重用户的隐私和体验。本文详细介绍了Android6.0运行时权限管理。需要的朋友一起来看下吧
    2016-12-12
  • Android中Dialog自定义上下文花式菜单

    Android中Dialog自定义上下文花式菜单

    这篇文章主要为大家详细介绍了Android中Dialog自定义上下文花式菜单,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • Android剪贴板用法详解

    Android剪贴板用法详解

    这篇文章主要介绍了Android剪贴板用法详解,以实例的形式对Android中剪贴板的各类传值方法做了较为详细的讲述,需要的朋友可以参考下
    2014-10-10
  • Android使用Handler实现定时器与倒计时器功能

    Android使用Handler实现定时器与倒计时器功能

    Handler的最常见应用场景之一便是通过Handler在子线程中间接更新UI。这篇文章主要介绍了Android使用Handler实现定时器与倒计时器功能,需要的朋友可以参考下
    2018-02-02
  • kotlin源码结构层次详解

    kotlin源码结构层次详解

    这篇文章主要为大家介绍了kotlin源码结构层次详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • ListView的View回收引起的checkbox状态改变监听等问题解决方案

    ListView的View回收引起的checkbox状态改变监听等问题解决方案

    之前讲到了自定义Adapter传递给ListView时,因为ListView的View回收,需要注意当ListView列表项中包含有带有状态标识控件的问题,感兴趣的朋友可以祥看本文,或许会有意外的收获哦
    2013-01-01
  • Android 实例开发基于ArcSoft实现人脸识别

    Android 实例开发基于ArcSoft实现人脸识别

    人脸识别,是基于人的脸部特征信息进行身份识别的一种生物识别技术。用摄像机或摄像头采集含有人脸的图像或视频流,并自动在图像中检测和跟踪人脸,进而对检测到的人脸进行识别的一系列相关技术,通常也叫做人像识别、面部识别
    2021-11-11
  • Android编程实现定时发短信功能示例

    Android编程实现定时发短信功能示例

    这篇文章主要介绍了Android编程实现定时发短信功能,结合实例形式较为详细的分析了Android定时发送短信功能的相关原理、实现方法与注意事项,需要的朋友可以参考下
    2017-09-09

最新评论