调用摄像头拍照并显示

本文介绍了一个简单的Android应用程序,该程序实现了拍照并裁剪图片的功能。文章详细展示了如何通过按钮点击事件触发相机拍摄,并将拍摄的照片保存到SD卡上指定的位置。此外,还介绍了如何启动系统的图片裁剪功能,以及如何将裁剪后的图片显示在ImageView中。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

package com.example.choosepictest;

import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Environment;
import android.provider.MediaStore;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.ImageButton;
import android.widget.ImageView;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;

public class MainActivity extends AppCompatActivity {

    public static  final  int TAKE_PHOTO = 1;

    public  static final int CROP_PHOTO = 2 ;

    private Button takephoto;

    private ImageView picture ;

    private Uri imageUri ;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        takephoto = (Button) findViewById(R.id.take_phone);
        picture = (ImageView) findViewById( R.id.picture) ;
        takephoto.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //创建File对象,用于存储拍照后的照片
                File outputImage = new File(Environment.getExternalStorageDirectory(),"output_image.jpg");
                System.out.println(Environment.getExternalStorageDirectory());
                try {
                    if(outputImage.exists()){
                        outputImage.delete();
                    }
                    outputImage.createNewFile();
                }catch (IOException e){
                    e.printStackTrace();
                }
                imageUri = Uri.fromFile(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, Intent data) {
        switch (requestCode){
            case TAKE_PHOTO:
                if(resultCode == RESULT_OK){
                    Intent intent = new Intent("com.android.camera.action.CROP");//为系统剪裁的Action

                    intent.setDataAndType(imageUri ,"image/*"); //设置剪裁的属性类型
                    intent.putExtra("scale",true);//scale为缩放的意思

                    intent.putExtra(MediaStore.EXTRA_OUTPUT , imageUri);
                    startActivityForResult(intent , CROP_PHOTO); //需要返回结果来判断是否截图成功的故用这个方法

                }
                break;
            case CROP_PHOTO:
                if(resultCode==RESULT_OK){
                    try{

                        Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));//而contentresolver则是用于管理所有程序的contentprovider实例
                        picture.setImageBitmap(bitmap);
                    }catch (FileNotFoundException e){
                        e.printStackTrace();
                    }
                }
                break;
            default:break;
        }
    }
}

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
   >
<Button
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:text="Take photo"
    android:id="@+id/take_phone"/>
    <ImageView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/picture"
        android:layout_gravity="center_vertical"/>
</LinearLayout>

总结:1 Environment.getExternalStorageDirectory() 得到SD卡的根目录
2 剪裁照片的属性设置
3 解析成位图

### Android 调用相机 API 拍照预览图片 为了实现在安卓应用中调用摄像头进行拍照将拍摄的照片显示出来,可以采用两种主要方法之一:使用 `Intent` 启动系统相机应用程序或直接通过 CameraX 或旧版 Camera API 控制硬件。 #### 方法一:启动系统相机应用程序 这种方法简单易行,适合大多数应用场景。下面是一个完整的例子来展示如何实现这一功能: ```java // 创建文件用于存储照片 File photoFile = createImageFile(); // 获取 URI 以便安全地传递给其他应用 Uri photoURI = CheckUtil.getUriForFile(this, photoFile); // 设置 Intent 动作以打开相机应用 Intent takePictureIntent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); takePictureIntent.putExtra(MediaStore.EXTRA_OUTPUT, photoURI); if (takePictureIntent.resolveActivity(getPackageManager()) != null) { startActivityForResult(takePictureIntent, REQUEST_TAKE_PHOTO); } ``` 这段代码创建了一个意图 (`Intent`) 来请求启动设备上的默认相机应用程序,指定了一个位置用来保存拍下的图像[^2]。 当用户完成拍照后,可以通过重写 `onActivityResult()` 方法处理返回的结果,在这里可以从指定的位置加载刚刚捕获到的图像数据将其设置在一个 `ImageView` 中供查看: ```java @Override protected void onActivityResult(int requestCode, int resultCode, @Nullable Intent data) { super.onActivityResult(requestCode, resultCode, data); if (requestCode == REQUEST_TAKE_PHOTO && resultCode == RESULT_OK) { // 加载图片至 ImageView Bitmap bitmap = BitmapFactory.decodeFile(photoFile.getAbsolutePath()); imageView.setImageBitmap(bitmap); } } ``` #### 方法二:使用 Camera1 API 进行拍照 对于更复杂的场景,可能需要更加精细地控制相机的行为,则可以直接利用 Camera1 API 完成同样的任务。以下是基于提供的参考资料中的片段扩展而成的一个简化版本的例子: ```kotlin private lateinit var camera: Camera fun capturePhoto() { val pictureCallback = object : PictureCallback { override fun onPictureTaken(data: ByteArray?, camera: Camera?) { // 将字节数组转换为位图对象 val options = BitmapFactory.Options() val imageBitmap = BitmapFactory.decodeByteArray(data, 0, data.size, options) runOnUiThread { // 更新 UI 线程上显示该图片 imageView.setImageBitmap(imageBitmap) // 停止预览释放资源 stopPreviewAndReleaseCamera() } // 取消自动聚焦模式恢复连续对焦状态 camera?.cancelAutoFocus() // 开始新的预览会话准备下一次拍摄 startPreview(surfaceHolder) } } // 请求捕捉一张静态照片 camera.takePicture(null, null, pictureCallback) } /// ... 其他必要的初始化逻辑 ... ``` 上述 Kotlin 函数展示了如何配置回调函数接收来自相机模块的数据流,进而获取所摄取的画面作为位图形式,最终呈现在界面上[^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值