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


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

     

     

     

     

     

     

     

     

  • 相关阅读:
    webSocket
    Spring中注解大全和应用
    原生js替换jQuery各种方法-中文版
    正则表达式-基础知识Review
    Swoole 4.1.0 正式版发布,支持原生 Redis/PDO/MySQLi 协程化
    Javascript 模块化指北
    精读《sqorn 源码》
    nodejs源码—初始化
    ubuntu显卡驱动安装
    iOS推断当前控制器是否在显示
  • 原文地址:https://www.cnblogs.com/xiaobaibailongma/p/16750089.html
Copyright © 2020-2023  润新知