多媒体——图片——使用相机拍摄图片

 

 

 

 

手机拍照的编码实现主要有两种:
(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());
    }

}

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

posted @ 2022-10-03 10:11  小白龙白龙马  阅读(115)  评论(0编辑  收藏  举报