目录
一、实验内容
通过编写代码实现打开相机拍照功能以及打开相册选择照片功能。
二、实验过程以及结果
1、首先创建界面布局,布局中包含两个Button(分别是btnTakePhoto和btnOpenPhoto)和一个ImageView,分别用于打开相机,打开相册以及展示图片。然后最重要的是要在 AndroidManifest.xml中声明权限,界面布局和权限声明如下。
2、为按钮btnTakePhoto设置点击事件。首先创建一个File对象用于存储拍摄后的照片,接着对运行设备的版本进行判断,当版本高于24或者低于24时分别采用不同方法获取imageUri。当版本高于24时需要调用FileProvider的getUriForFile()方法将File对象转换成一个封装过的Uri对象,而FileProvider 是一种特殊的内容提供器,那么我们自然要在AndroidManifest.xml中对内容提供器进行注册。最后最重要的是进行拍照时权限的申请。下面是点击事件中的代码以及内容提供器的注册。
<provider
android:authorities="com.example.lab.mymultimedia.fileprovider"
android:name="android.support.v4.content.FileProvider"
android:exported="false"
android:grantUriPermissions="true">
<meta-data
android:name="android.support.FILE_PROVIDER_PATHS"
android:resource="@xml/file_paths" />
</provider>
3、在上一步内容提供器的注册中,android:name属性的值是固定的,android:authorities属性的值必须要和刚才FileProvider.getUriForFile()方法中的第二个参数一致。另外,这里还在<provider>标签的内部使用<meta-data>来指定Uri的共享路径,并引用了一个@xml/file_paths资源,现在这个资源是不存在的,下面我们就来创建它。
右击res目录→New→Directory,创建一个xml目录,接着右击xml目录→New→File,创建一个file_paths.xml文件。然后修改file_paths.xml中的内容,如下。
<?xml version="1.0" encoding="utf-8"?>
<resources>
<paths>
<external-path
name="camera_photos"
path="" /> <!--定义共享文件路径,空表示整个sd卡-->
</paths>
</resources>
4、为按钮btnOpenPhoto设置点击事件。直接在打开相册时进行申请权限即可。
5、因为步骤2、4代码中都有进行权限申请,用户可在界面中通过弹窗自由选择允许或者拒绝权限,所以接下来要对用户权限的选择进行处理。若授权则启动相机或者打开相册,否则用Toast弹出无权限的提示。
6、下面是启动相机和打开相册的代码。两者都是用Intent打开活动,并在最后都使用startActivityForResult()来等待数据返回。
7、因为之前使用了startActivityForResult()方法,所以在onActivityResult()中会得到返回结果。若返回结果为RESULT_OK,则在ImageView中显示刚拍摄的照片或者处理刚才在相册中 选中的照片。
8、上一步中处理相册选中照片要根据手机版本号使用不同方法处理,若版本大于等于19,则使用handleImageOnKitKat()方法,否则使用handleImageBeforeKitKat()方法,下面分别是两种方法的代码。
9、下面分别为获取图片真实路径以及展示相册选中图片的方法。
10、结果展示。由于电脑没有摄像头,图中的拍摄背景是模拟器自带的虚拟拍摄背景。
三、代码
MainActivity .java
public class MainActivity extends AppCompatActivity {
//设置请求码
public static final int REQUEST_CAMERA=1;
public static final int CHOOSE_PHOTO=2;
private Uri imageUri;
//用于显示图片
private ImageView IvPhoto;
//用于存储拍照后的图像
private File file;
Intent intent;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//实例化各控件
IvPhoto=(ImageView)findViewById(R.id.imageView);
Button btnTakePhoto=(Button)findViewById(R.id.take_photo);
Button btnOpenPhoto=(Button)findViewById(R.id.open_photo);
//设置拍照点击事件
btnTakePhoto.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//创建File对象,用于存储拍照后的图片
file = new File(getExternalCacheDir(), "photo.jpg");
try{
//若文件存在,则删除再重新创建
if(file.exists()){
file.delete();
}
file.createNewFile();
}catch(IOException e){
e.printStackTrace();
}
if(Build.VERSION.SDK_INT>=24){
//若运行设备的系统版本高于7.0(API24),则用此方法获取imageUri
imageUri = FileProvider.getUriForFile(MainActivity.this,
"com.example.lab.mymultimedia.fileprovider", file);
}else{
//若运行设备的系统版本高于7.0(API24),此方法获取imageUri将报错
imageUri=Uri.fromFile(file);
}
//运行拍照时权限申请
if(ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.
CAMERA) != PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(MainActivity.this,
new String[]{Manifest.permission.CAMERA},1);
}else{
//启动相机程序
openCamera();
}
}
});
//相册点击事件
btnOpenPhoto.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
//打开相册时权限申请
if(ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.
WRITE_EXTERNAL_STORAGE) != PackageManager.PERMISSION_GRANTED){
ActivityCompat.requestPermissions(MainActivity.this,
new String[]{Manifest.permission.WRITE_EXTERNAL_STORAGE},2);
}else{
//打开相册
openAlbum();
}
}
});
}
//启动摄像机
private void openCamera(){
intent = new Intent("android.media.action.IMAGE_CAPTURE");
ContentValues contentValues = new ContentValues(1);
contentValues.put(MediaStore.Images.Media.DATA, file.getAbsolutePath());
//调用Intent的putExtra()方法指定图片的输出地址
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, REQUEST_CAMERA);
}
//打开相册
private void openAlbum(){
//打开通用界面,包括了音频、图片等。
Intent intent=new Intent("android.intent.action.GET_CONTENT");
//过滤,只显示图片
intent.setType("image/*");
startActivityForResult(intent,CHOOSE_PHOTO);
}
//对用户权限的选择进行处理
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
switch (requestCode){
case 1:
if(grantResults.length>0 && grantResults[0]==PackageManager.PERMISSION_GRANTED ){
//启动相机程序
openCamera();
}else{
Toast.makeText(this,"你没有拍照的权限!",Toast.LENGTH_SHORT).show();
}
case 2:
if(grantResults.length>0 && grantResults[0]==PackageManager.PERMISSION_GRANTED ){
//打开相册
openAlbum();
}else{
Toast.makeText(this,"你没有打开相册的权限!",Toast.LENGTH_SHORT).show();
}
break;
default:
}
}
//因为之前使用startActivityForResult()来启动活动,所以会有结果返回在onActivityResult()中。
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
switch(requestCode){
case REQUEST_CAMERA:
if(resultCode==RESULT_OK){
try{
//将刚才拍摄的图片显示出来
Bitmap bitmap= BitmapFactory.decodeStream(getContentResolver().
openInputStream(imageUri));
IvPhoto.setImageBitmap(bitmap);
}catch(FileNotFoundException e){
e.printStackTrace();
}
}
break;
case CHOOSE_PHOTO:
if(resultCode==RESULT_OK)
{
//判断手机版本号
if(Build.VERSION.SDK_INT>=19){
//4.4及以上使用该方法处理图片
handleImageOnKitKat(data);
}else{
//4.4以下使用该方法处理图片
handleImageBeforeKitKat(data);
}
}
break;
default:
break;
}
}
@TargetApi(19)
private void handleImageOnKitKat(Intent data){
String imagePath=null;
Uri uri=data.getData();
if(DocumentsContract.isDocumentUri(this,uri)){
//如果是document类型的Uri,则通过document id处理
String docId=DocumentsContract.getDocumentId(uri);
if("com.android.providers.media.documents".equals(uri.getAuthority())){
String id=docId.split(":")[1];//解析出数字格式的id
String selection=MediaStore.Images.Media._ID + "=" +id;
//获得选中图片的路径
imagePath=getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI,selection);
}else if("com.android.providers.dowloads.documents".equals(uri.getAuthority())){
Uri contentUri= ContentUris.withAppendedId(Uri.parse("content:" +
"//downloads/public_downloads"),Long.valueOf(docId));
//获得选中图片的路径
imagePath=getImagePath(contentUri,null);
}
}else if("content".equalsIgnoreCase(uri.getScheme())){
//若是content类型的Uri,则使用普通方式处理获得选中图片的路径
imagePath=getImagePath(uri,null);
}else if("file".equalsIgnoreCase(uri.getScheme())){
//若是file类型的Uri,则直接获得选中图片的路径
imagePath=uri.getPath();
}
//显示选中图片
displayImage(imagePath);
}
private void handleImageBeforeKitKat(Intent data){
Uri uri=data.getData();
String 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);
IvPhoto.setImageBitmap(bitmap);
}else{
Toast.makeText(this,"打开照片失败!",Toast.LENGTH_SHORT).show();
}
}
}