多媒体——图片——使用相机拍摄图片
手机拍照的编码实现主要有两种:
(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()); } }