AS2022简单实现调用手机相机拍照并显示照片

需求:通过点击拍照按钮来调用手机相机,拍照后显示在屏幕中

1.布局文件设计

xml文件中放置一个按钮以及一个imageView控件(图片放置于此)

PS:个人推荐直接用约束布局拖组件

代码如下:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent">

    <Button
        android:id="@+id/take_photo"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="拍照"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintHorizontal_bias="0.498"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintVertical_bias="0.109" />

    <ImageView
        android:id="@+id/picture"
        android:layout_width="0dp"
        android:layout_height="0dp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintHorizontal_bias="0.461"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="@+id/guideline2"
        tools:srcCompat="@drawable/baseline_child_care_24" />

    <androidx.constraintlayout.widget.Guideline
        android:id="@+id/guideline2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:orientation="horizontal"
        app:layout_constraintGuide_percent="0.5" />
</androidx.constraintlayout.widget.ConstraintLayout>

2.授予权限

由于我们实现的功能是在公共存储读写数据,就必须使用“WRITE_EXTERNAL_STORAGE”权限,也就要去掉“ android:maxSdkVersion="18"”。

在manifest文件中授权

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

3.使用FIleProvider类

由于我们的需求需要在我们当前应用程序和相机之间共享数据(即所拍照片),我们需要用到FileProvider类

配置过程:

首先需要在AndroidManifest.xml文件中通过<provider>标签进行声明,并配置自定义android:authorities属性,代码如下:

<manifest>
    <application>
        <provider
            android:name="android.support.v4.content.FileProvider"
            android:authorities="my.itgungnir.fileprovider"
            android:exported="false"
            android:grantUriPermissions="true">
        </provider>
    </application>
</manifest>

如果你的模拟器版本在Android10之下,就用android.support.v4.content.FileProvider

为了让其他应用可以访问当前应用下的文件,我们还需要配置哪些文件夹可以被访问,这个步骤也是在XML文件中配置的。我们需要在项目的/res/xml文件夹下创建一个路径配置文件,命名为file_paths.xml(文件名可以自定义),这个文件中的根节点是<paths>,在这个节点下配置文件夹。一个示例配置如下:

<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
    <external-path name="my_images" path="/"/>
</paths>

path路径设为“/”,表示能够访问所有文件夹

配置完各个文件夹之后,我们还需要在AndroidManifest.xml文件中的<provider>标签中引用这个配置,这样才能使配置生效。使用<meta-data>标签来引用配置,代码如下:

<provider
            android:authorities="Tom_LZK"
            android:name="androidx.core.content.FileProvider"
            android:exported="false"
            android:grantUriPermissions="true">
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            android:resource="@xml/file_paths"/>
        </provider>

<meta-data>标签中的android:name属性是固定值android.support.FILE_PROVIDER_PATHSandroid:resource属性是对上面的配置文件的引用。

附上MainActivity的代码:

package com.example.arbitrary;

import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Bundle;
import android.provider.MediaStore;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;

import androidx.annotation.Nullable;
import androidx.appcompat.app.AppCompatActivity;
import androidx.core.content.FileProvider;

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

public class MainActivity extends AppCompatActivity {
    public static final int TAKE_PHOTO = 1;
    private ImageView picture;
    private Uri imageUri;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Button takePhoto = findViewById(R.id.take_photo);
        picture = findViewById(R.id.picture);
        takePhoto.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                //创建File对象,用来存储拍照后的照片
                //getExternalCacheDir()获取此应用缓存数据的位置,在这个位置保存图片
                File outputImage = new File(getExternalCacheDir(), "output_image.jpg");
                try {
                    if (outputImage.exists()) {//如果图片已经存在就删除再重新创建
                        outputImage.delete();
                    }
                    outputImage.createNewFile();
                } catch (IOException e) {
                    e.printStackTrace();
                }
                imageUri = FileProvider.getUriForFile(MainActivity.this,
                        "Tom_LZK", outputImage);

                //启动相机
                Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
                intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
                startActivityForResult(intent, TAKE_PHOTO);
            }
        });
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (requestCode == TAKE_PHOTO && resultCode == RESULT_OK) {
                    //显示拍出来的照片
                    Bitmap bitmap = BitmapFactory.decodeFile(getExternalCacheDir()+"//output_image.jpg");
                    picture.setImageBitmap(bitmap);
                }
        }
}

完成。 

注意:getUriForFile方法中authorities变量要与provider中的authorities一致

猜你喜欢

转载自blog.csdn.net/weixin_64089259/article/details/129813180