热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android实现调用摄像头和相册的方法

这篇文章主要为大家详细介绍了Android实现调用摄像头和相册的方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

Android调用摄像头是很方便的。先看一下界面

布局文件activity_main.xml源码

<&#63;xml version="1.0" encoding="utf-8"&#63;> 
 
 
  

因为涉及到向SD卡写入数据,所有需要在AndroidMainfest.xml中声明响应权限

 

MainActivity.java源码

package com.example.luoxn28.activity; 
 
import android.annotation.TargetApi; 
import android.content.ContentUris; 
import android.content.Intent; 
import android.database.Cursor; 
import android.graphics.Bitmap; 
import android.graphics.BitmapFactory; 
import android.net.Uri; 
import android.os.Bundle; 
import android.os.Environment; 
import android.provider.DocumentsContract; 
import android.provider.MediaStore; 
import android.support.v7.app.ActionBarActivity; 
import android.view.View; 
import android.widget.Button; 
import android.widget.ImageView; 
import android.widget.Toast; 
 
import java.io.File; 
import java.io.FileNotFoundException; 
import java.io.IOException; 
 
public class MainActivity extends ActionBarActivity { 
  private static final String TAG = "hdu"; 
 
  public static final int TAKE_PHOTO = 1; 
  public static final int CROP_PHOTO = 2; 
  public static final int CHOOSE_PHOTO = 3; 
 
  private Button takePhoto; 
  private ImageView picture; 
  private Uri imageUri; 
 
  private Button chooseFromAlbum; 
 
  @Override 
  protected void onCreate(Bundle savedInstanceState) { 
    super.onCreate(savedInstanceState); 
    //requestWindowFeature(Window.FEATURE_NO_TITLE); 
    setContentView(R.layout.activity_main); 
 
    takePhoto = (Button) findViewById(R.id.take_photo); 
    picture = (ImageView) findViewById(R.id.picture); 
    chooseFromAlbum = (Button) findViewById(R.id.choose_from_album); 
 
    takePhoto.setOnClickListener(new View.OnClickListener() { 
      @Override 
      public void onClick(View v) { 
        // 创建File对象,用于存储拍摄后照片 
        File saveImage = new File(Environment.getExternalStorageDirectory(), "saveImage.jpg"); 
        try { 
          if (saveImage.exists()) { 
            saveImage.delete(); 
          } 
          saveImage.createNewFile(); 
        } 
        catch (IOException ex) { 
          ex.printStackTrace(); 
        } 
 
        imageUri = Uri.fromFile(saveImage); 
        Intent intent = new Intent("android.media.action.IMAGE_CAPTURE"); 
        intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri); 
        // 启动相机 
        startActivityForResult(intent, TAKE_PHOTO); 
      } 
    }); 
 
    chooseFromAlbum.setOnClickListener(new View.OnClickListener() { 
      @Override 
      public void onClick(View v) { 
        Intent intent = new Intent("android.intent.action.GET_CONTENT"); 
        intent.setType("image/*"); 
        // 打开相册 
        startActivityForResult(intent, CHOOSE_PHOTO); 
      } 
    }); 
  } 
 
  @Override 
  protected void onActivityResult(int requestCode, int resultCode, Intent data) { 
    switch (requestCode) { 
      case TAKE_PHOTO: 
        if (resultCode == RESULT_OK) { 
          Intent intent = new Intent("com.android.camera.action.CROP"); 
          intent.setDataAndType(imageUri, "image/*"); 
          intent.putExtra("scale", true); 
          intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri); 
          // 启动裁剪程序 
          startActivityForResult(intent, CROP_PHOTO); 
        } 
        break; 
 
      case CROP_PHOTO: 
        if (resultCode == RESULT_OK) { 
          try { 
            Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri)); 
            // 显示裁剪后的图片 
            picture.setImageBitmap(bitmap); 
          } 
          catch (FileNotFoundException ex) { 
            ex.printStackTrace(); 
          } 
        } 
        break; 
 
      case CHOOSE_PHOTO: 
        if (resultCode == RESULT_OK) { 
          handleImage(data); 
        } 
        break; 
 
      default: 
        break; 
    } 
  } 
 
  // 只在Android4.4及以上版本使用 
  @TargetApi(19) 
  private void handleImage(Intent data) { 
    String imagePath = null; 
    Uri uri = data.getData(); 
 
    if (DocumentsContract.isDocumentUri(this, uri)) { 
      // 通过document id来处理 
      String docId = DocumentsContract.getDocumentId(uri); 
      if ("com.android.providers.media.documents".equals(uri.getAuthority())) { 
        // 解析出数字id 
        String id = docId.split(":")[1]; 
        String selection = MediaStore.Images.Media._ID + "=" + id; 
        imagePath = getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, selection); 
      } 
      else if ("com.android.providers.downloads.documents".equals(uri.getAuthority())) { 
        Uri cOntentUri= ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"), 
            Long.valueOf(docId)); 
        imagePath = getImagePath(contentUri, null); 
      } 
    } 
    else if ("content".equals(uri.getScheme())) { 
      // 如果不是document类型的Uri,则使用普通方式处理 
      imagePath = getImagePath(uri, null); 
    } 
 
    // 根据图片路径显示图片 
    displayImage(imagePath); 
  } 
 
  private String getImagePath(Uri uri, String selection) { 
    String path = null; 
    // 通过Uri和selection来获取真实图片路径 
    Cursor cursor = getContentResolver().query(uri, null, selection, null, null); 
    if (cursor != null) { 
      if (cursor.moveToFirst()) { 
        path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA)); 
      } 
      cursor.close(); 
    } 
 
    return path; 
  } 
 
  private void displayImage(String imagePath) { 
    if (imagePath != null) { 
      Bitmap bitmap = BitmapFactory.decodeFile(imagePath); 
      picture.setImageBitmap(bitmap); 
    } 
    else { 
      Toast.makeText(this, "failed to get image", Toast.LENGTH_SHORT).show(); 
    } 
  } 
} 

调用摄像头拍照

       在MainActivity 中要做的第一件事自然是分别获取到 Button 和 ImageView 的实例,并给 Button 注册上点击事件,然后在 Button的点击事件里开始处理调用摄像头的逻辑,我们重点看下这部分代码。
       首先这里创建了一个 File 对象,用于存储摄像头拍下的图片,这里我们把图片命名为saveImage.jpg ,并将它存放在手机SD卡的根目录下,调 用 Environment 的getExternalStorageDirectory()方法获取到的就是手机 SD 卡的根目录。然后再调用 Uri 的fromFile()方法将 File 对象转换成 Uri 对象,这个 Uri 对象标识着 saveImage.jpg 这张图片的唯一地址。 接着构建出一个 Intent对象, 并将这个 Intent的 action指定为android.media.action.IMAGE_CAPTURE,再调用 Intent 的 putExtra()方法指定图片的输出地址,这里填入刚刚得到的 Uri 对象,最后调用 startActivityForResult()来启动活动。由于我们使用的是一个隐式Intent,系统会找出能够响应这个 Intent 的活动去启动,这样照相机程序就会被打开,拍下的照片将会输出到 saveImage.jpg 中。

        注意刚才我们是使用 startActivityForResult()来启动活动的,因此拍完照后会有结果返回到 onActivityResult()方法中。如果发现拍照成功,则会再次构建出一个 Intent 对象,并把它的 action 指定为 com.android.camera.action.CROP。这个 Intent 是用于对拍出的照片进行裁剪注意刚才我们是使用 startActivityForResult()来启动活动的,因此拍完照后会有结果返回到 onActivityResult()方法中。如果发现拍照成功,则会再次构建出一个 Intent 对象,并把它的 action 指定为 com.android.camera.action.CROP。这个 Intent 是用于对拍出的照片进行裁剪

从相册中选择照片

        在 "从相册中选择图片"按钮的点击事件里我们同样创建了一个 File 对象,用于存储从相册中选择的图片。然后构建出一个 Intent 对象,并将它的 action 指定为android.intent.action.GET_CONTENT。接着给这个 Intent 对象设置一些必要的参数,包括是否允许缩放和裁剪、图片的输出位置等。最后调用 startActivityForResult()方法,就可以打开相册程序选择照片了。

        注意在调用 startActivityForResult()方法的时候,我们给第二个参数传入的值仍然是CROP_PHOTO 常量,这样的好处就是从相册选择好照片之后,会直接进入到 CROP_PHOTO的 case 下将图片显示出来, 这样就可以复用之前写好的显示图片的逻辑, 不用再编写一遍了。

参考资料

        1、《第一行代码-Android》调用摄像头章节

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


推荐阅读
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 国内BI工具迎战国际巨头Tableau,稳步崛起
    尽管商业智能(BI)工具在中国的普及程度尚不及国际市场,但近年来,随着本土企业的持续创新和市场推广,国内主流BI工具正逐渐崭露头角。面对国际品牌如Tableau的强大竞争,国内BI工具通过不断优化产品和技术,赢得了越来越多用户的认可。 ... [详细]
  • 本文探讨了 RESTful API 和传统接口之间的关键差异,解释了为什么 RESTful API 在设计和实现上具有独特的优势。 ... [详细]
  • 本文详细介绍了如何使用Spring Boot进行高效开发,涵盖了配置、实例化容器以及核心注解的使用方法。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • Python自动化处理:从Word文档提取内容并生成带水印的PDF
    本文介绍如何利用Python实现从特定网站下载Word文档,去除水印并添加自定义水印,最终将文档转换为PDF格式。该方法适用于批量处理和自动化需求。 ... [详细]
  • 在当前众多持久层框架中,MyBatis(前身为iBatis)凭借其轻量级、易用性和对SQL的直接支持,成为许多开发者的首选。本文将详细探讨MyBatis的核心概念、设计理念及其优势。 ... [详细]
  • 将Web服务部署到Tomcat
    本文介绍了如何在JDeveloper 12c中创建一个Java项目,并将其打包为Web服务,然后部署到Tomcat服务器。内容涵盖从项目创建、编写Web服务代码、配置相关XML文件到最终的本地部署和验证。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • 本文介绍如何在 Unity 的 XML 配置文件中,将参数传递给自定义生命周期管理器的构造函数。我们将详细探讨 CustomLifetimeManager 类的实现及其配置方法。 ... [详细]
  • 本文详细介绍了 Java 中 org.apache.xmlbeans.SchemaType 类的 getBaseEnumType() 方法,提供了多个代码示例,并解释了其在不同场景下的使用方法。 ... [详细]
  • 本文详细介绍了如何解决MyBatis中常见的BindingException错误,提供了多种排查和修复方法,确保Mapper接口与XML文件的正确配置。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
author-avatar
用户q4oaa53j5h
这个家伙很懒,什么也没留下!
RankList | 热门文章
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有