Requisito: Ligue para a câmera do telefone clicando no botão da câmera e exiba-a na tela depois de tirar uma foto
1. Projeto de arquivo de layout
Coloque um botão e um controle imageView no arquivo xml (a imagem é colocada aqui)
PS: Eu pessoalmente recomendo arrastar os componentes diretamente com o layout de restrição
código mostra como abaixo:
<?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. Conceder permissões
Como a função que implementamos é ler e gravar dados no armazenamento público, devemos usar a permissão "WRITE_EXTERNAL_STORAGE" e devemos remover "android:maxSdkVersion="18"".
Autorize no arquivo de manifesto
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
android:maxSdkVersion="18"/>
3. Use a classe FILeProvider
Como nossas necessidades precisam compartilhar dados (ou seja, fotos tiradas) entre nosso aplicativo atual e a câmera, precisamos usar a classe FileProvider
Processo de configuração:
Primeiro de tudo , você precisa declarar
AndroidManifest.xml
através de tags no arquivo<provider>
e configurar atributos personalizadosandroid:authorities
. O código é o seguinte:
<manifest>
<application>
<provider
android:name="android.support.v4.content.FileProvider"
android:authorities="my.itgungnir.fileprovider"
android:exported="false"
android:grantUriPermissions="true">
</provider>
</application>
</manifest>
Se a versão do seu emulador for inferior a Android10, use android.support.v4.content.FileProvider
Para permitir que outros aplicativos acessem os arquivos do aplicativo atual, também precisamos configurar quais pastas podem ser acessadas, esta etapa também é XML
configurada no arquivo. Precisamos criar um arquivo de configuração de caminho na /res/xml
pasta do projeto, nomeá-lo file_paths.xml
(o nome do arquivo pode ser personalizado) e o nó raiz neste arquivo é <paths>
configurar a pasta sob este nó. Um exemplo de configuração é o seguinte:
<?xml version="1.0" encoding="utf-8"?>
<paths xmlns:android="http://schemas.android.com/apk/res/android">
<external-path name="my_images" path="/"/>
</paths>
O path path está definido como "/", o que significa que todas as pastas podem ser acessadas
Depois de configurar cada pasta, também precisamos referenciar essa configuração no rótulo AndroidManifest.xml
do arquivo <provider>
, para que a configuração tenha efeito. Use <meta-data>
tags para se referir à configuração, o código é o seguinte:
<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>
Os atributos nas tags android:name
são valores fixos android.support.FILE_PROVIDER_PATHS
e android:resource
os atributos são referências aos arquivos de configuração acima.
Anexe o código da 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);
}
}
}
Terminar.
Observação: a variável de autoridades no método getUriForFile deve ser consistente com as autoridades no provedor