手机拍照的编码实现主要有两种:
(1)通过Camera工具联合表面视图SurfaceView,由开发者实现拍照细节;
(2)借助系统相机自动拍照,也就是跳到系统相机页面,由系统相机拍摄照片;
Intent photoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); // 下面通过系统相机拍照只能获得缩略图
startActivityForResult(photoIntent, THUMBNAIL_CODE); // 打开系统相机
重写系统相机的回调方法
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent intent)
{
super.onActivityResult(requestCode, resultCode, intent);
if (resultCode==RESULT_OK && requestCode==THUMBNAIL_CODE)// 获得缩略图
{ // 缩略图放在返回意图中的data字段,将其取出转成位图对象即可
Bundle extras = intent.getExtras();
Bitmap bitmap = (Bitmap) extras.get("data");
iv_photo.setImageBitmap(bitmap); // 设置图像视图的位图对象
}
}
ContentValues values = new ContentValues(); // Android10开始必须由系统自动分配路径,同时该方式也能自动刷新相册
values.put(MediaStore.Video.Media.DISPLAY_NAME, "photo_"+DateUtil.getNowDateTime()); // 指定图片文件的名称
values.put(MediaStore.Video.Media.MIME_TYPE, "image/jpeg"); // 类型为图像
mImageUri = getContentResolver().insert(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values); // 通过内容解析器插入一条外部内容的路径信息
photoIntent.putExtra(MediaStore.EXTRA_OUTPUT, mImageUri); // 下面通过系统相机拍照可以获得原始图
startActivityForResult(photoIntent, ORIGINAL_CODE); // 打开系统相机
布局:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal">
<Button
android:id="@+id/btn_thumbnail"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="打开相机(缩略图)"
android:textColor="@color/black"
android:textSize="17sp" />
<Button
android:id="@+id/btn_original"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_weight="1"
android:text="打开相机(原始照片)"
android:textColor="@color/black"
android:textSize="17sp" />
</LinearLayout>
<ImageView
android:id="@+id/iv_photo"
android:layout_width="match_parent"
android:layout_height="360dp"
android:scaleType="fitCenter" />
</LinearLayout>
权限:
<!-- 相机 -->
<uses-permission android:name="android.permission.CAMERA" />
<!-- 录音 -->
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<!-- 存储卡读写 -->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MANAGE_EXTERNAL_STORAGE" />
主代码:
package com.example.myapplication;
import android.content.ContentValues;
import android.content.Intent;
import android.graphics.Bitmap;
import android.net.Uri;
import android.os.Bundle;
import android.provider.MediaStore;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;
import androidx.appcompat.app.AppCompatActivity;
import com.example.myapplication.util.BitmapUtil;
import com.example.myapplication.util.DateUtil;
public class MainActivity extends AppCompatActivity implements View.OnClickListener
{
private final static String TAG = "PhotoTakeActivity";
private int THUMBNAIL_CODE = 1; // 获取缩略图的请求码
private int ORIGINAL_CODE = 2; // 获取原始图的请求码
private ImageView iv_photo; // 声明一个图像视图对象
private Uri mImageUri; // 图片的路径对象
@Override
protected void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
iv_photo = findViewById(R.id.iv_photo);
findViewById(R.id.btn_thumbnail).setOnClickListener(this);
findViewById(R.id.btn_original).setOnClickListener(this);
}
@Override
public void onClick(View v)
{
if (v.getId() == R.id.btn_thumbnail)
{
// 下面通过系统相机拍照只能获得缩略图
Intent photoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
startActivityForResult(photoIntent, THUMBNAIL_CODE); // 打开系统相机
}
else if (v.getId() == R.id.btn_original)
{
takeOriginalPhoto(); // 拍照时获取原始图片
}
}
// 拍照时获取原始图片
private void takeOriginalPhoto()
{
Intent photoIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
// Android10开始必须由系统自动分配路径,同时该方式也能自动刷新相册
ContentValues values = new ContentValues();
// 指定图片文件的名称
values.put(MediaStore.Video.Media.DISPLAY_NAME, "photo_"+DateUtil.getNowDateTime());
values.put(MediaStore.Video.Media.MIME_TYPE, "image/jpeg"); // 类型为图像
// 通过内容解析器插入一条外部内容的路径信息
mImageUri = getContentResolver().insert(
MediaStore.Images.Media.EXTERNAL_CONTENT_URI, values);
// if (Build.VERSION.SDK_INT < Build.VERSION_CODES.Q) { // 不推荐使用以下代码,因为不会自动刷新相册
// // 获得图片的临时保存路径
// String filePath = String.format("%s/%s.jpg",
// getExternalFilesDir(Environment.DIRECTORY_PICTURES), "photo_"+ DateUtil.getNowDateTime());
// if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { // 7.0以上要通过FileProvider转换
// mImageUri = FileProvider.getUriForFile(this, getString(R.string.file_provider), new File(filePath));
// } else { // 7.0以下直接根据路径生成对应的Uri
// mImageUri = Uri.parse(filePath);
// }
// }
// 下面通过系统相机拍照可以获得原始图
photoIntent.putExtra(MediaStore.EXTRA_OUTPUT, mImageUri);
startActivityForResult(photoIntent, ORIGINAL_CODE); // 打开系统相机
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent intent)
{
super.onActivityResult(requestCode, resultCode, intent);
if (resultCode==RESULT_OK && requestCode==THUMBNAIL_CODE) // 获得缩略图
{
// 缩略图放在返回意图中的data字段,将其取出转成位图对象即可
Bundle extras = intent.getExtras();
Bitmap bitmap = (Bitmap) extras.get("data");
iv_photo.setImageBitmap(bitmap); // 设置图像视图的位图对象
Log.d(TAG, "getWidth="+bitmap.getWidth()+", getHeight="+bitmap.getHeight());
}
if (resultCode==RESULT_OK && requestCode==ORIGINAL_CODE) // 获得原始图
{
//iv_photo.setLayerType(View.LAYER_TYPE_SOFTWARE, null); // 设置图层类型为软件加速
//iv_photo.setImageURI(mImageUri); // 设置图像视图的路径对象
// 需要自动缩小原始图片,因为过大的图片无法显示,会报下列错误:
// Bitmap too large to be uploaded into a texture (3120x4208, max=4096x4096)
// 根据指定图片的uri,获得自动缩小后的位图对象
Bitmap bitmap = BitmapUtil.getAutoZoomImage(this, mImageUri);
iv_photo.setImageBitmap(bitmap); // 设置图像视图的位图对象
Log.d(TAG, "getWidth="+bitmap.getWidth()+", getHeight="+bitmap.getHeight());
}
}
}
BitmapUtil
package com.example.myapplication.util;
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Matrix;
import android.net.Uri;
import android.util.Log;
import java.io.InputStream;
public class BitmapUtil
{
private final static String TAG = "BitmapUtil";
// 获得旋转角度之后的位图对象
public static Bitmap getRotateBitmap(Bitmap bitmap, float rotateDegree)
{
Matrix matrix = new Matrix(); // 创建操作图片用的矩阵对象
matrix.postRotate(rotateDegree); // 执行图片的旋转动作
// 创建并返回旋转后的位图对象
return Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(), bitmap.getHeight(), matrix, false);
}
// 获得比例缩放之后的位图对象
public static Bitmap getScaleBitmap(Bitmap bitmap, double scaleRatio)
{
int new_width = (int) (bitmap.getWidth() * scaleRatio);
int new_height = (int) (bitmap.getHeight() * scaleRatio);
// 创建并返回缩放后的位图对象
return Bitmap.createScaledBitmap(bitmap, new_width, new_height, false);
}
// 获得自动缩小后的位图对象
public static Bitmap getAutoZoomImage(Context ctx, Uri uri)
{
Log.d(TAG, "getAutoZoomImage uri="+uri.toString());
Bitmap zoomBitmap = null;
// 打开指定uri获得输入流对象
try (InputStream is = ctx.getContentResolver().openInputStream(uri))
{
// 从输入流解码得到原始的位图对象
Bitmap originBitmap = BitmapFactory.decodeStream(is);
int ratio = originBitmap.getWidth()/2000+1;
// 获得比例缩放之后的位图对象
zoomBitmap = BitmapUtil.getScaleBitmap(originBitmap, 1.0/ratio);
}
catch (Exception e)
{
e.printStackTrace();
}
return zoomBitmap;
}
}
DateUtil
package com.example.myapplication.util;
import android.annotation.SuppressLint;
import android.text.TextUtils;
import java.text.SimpleDateFormat;
import java.util.Date;
@SuppressLint("SimpleDateFormat")
public class DateUtil {
// 获取指定格式的日期时间
public static String getNowDateTime(String formatStr) {
String format = formatStr;
if (TextUtils.isEmpty(format)) {
format = "yyyyMMddHHmmss";
}
SimpleDateFormat sdf = new SimpleDateFormat(format);
return sdf.format(new Date());
}
// 获取当前的日期时间
public static String getNowDateTime() {
SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmss");
return sdf.format(new Date());
}
// 获取当前的时间
public static String getNowTime() {
SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss");
return sdf.format(new Date());
}
}