1、调用摄像头拍照
首先我们来实现拍照,然后把这张相片显示出来的应用。新建ChoosePicTest项目,然后修改布局如下:
<!-- activity_main.xml -->
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
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="拍照"
android:id="@+id/take_photo_button"/>
<ImageView android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/photo"
android:layout_gravity="center_horizontal"/>
</LinearLayout>
然后修改主函数:
上面的代码比较复杂,分析我就用第一行代码的原文了:
上述代码稍微有点复杂,我们来仔细地分析一下。在MainActivity 中要做的第一件事自然是分别获取到Button 和ImageView 的实例,并给Button 注册上点击事件,然后在Button的点击事件里开始处理调用摄像头的逻辑,我们重点看下这部分代码。
首先这里创建了一个File 对象,用于存储摄像头拍下的图片,这里我们把图片命名为output_image.jpg , 并将它存放在手机SD 卡的根目录下, 调用Environment 的getExternalStorageDirectory()方法获取到的就是手机SD 卡的根目录。然后再调用Uri 的
fromFile()方法将File 对象转换成Uri 对象,这个Uri 对象标识着output_image.jpg 这张图片的唯一地址。接着构建出一个Intent对象,并将这个Intent的action指定为android.media.action.
IMAGE_CAPTURE,再调用Intent 的putExtra()方法指定图片的输出地址,这里填入刚刚得到的Uri 对象,最后调用startActivityForResult()来启动活动。由于我们使用的是一个隐式
Intent,系统会找出能够响应这个Intent 的活动去启动,这样照相机程序就会被打开,拍下的照片将会输出到output_image.jpg 中。
注意刚才我们是使用startActivityForResult()来启动活动的,因此拍完照后会有结果返回到onActivityResult()方法中。如果发现拍照成功,则会再次构建出一个Intent 对象,并把它的action 指定为com.android.camera.action.CROP。这个Intent 是用于对拍出的照片进行裁剪的,因为摄像头拍出的照片都比较大,而我们可能只希望截取其中的一小部分。然后给这个Intent 设置上一些必要的属性,并再次调用startActivityForResult()来启动裁剪程序。裁剪后的照片同样会输出到output_image.jpg 中。
裁剪操作完成之后,程序又会回调到onActivityResult()方法中,这个时候我们就可以调用BitmapFactory 的decodeStream()方法将output_image.jpg 这张照片解析成Bitmap 对象,然后把它设置到ImageView 中显示出来。
最后是添加写入SD卡的权限:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
实验结果:
2、从相册中选择图片
这个功能在应用中也是很常见的,往往和前面拍照的功能放在一起。下面看看这个功能怎么实现。首先在布局中添加按钮:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
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="拍照"
android:id="@+id/take_photo_button"/>
<Button android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/choose_from_album_button"
android:text="从相册中选图片"/>
<ImageView android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/photo"
android:layout_gravity="center_horizontal"/>
</LinearLayout>
然后添加点击事件,图片选出之后调用的依然是裁剪图片的代码:
mBChooseFromAlbum.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View arg0) {
// 创建File对象,用来存储选择的图片
File outputImage = new File(Environment.getExternalStorageDirectory(), "output_image.jpg");
try {
if (outputImage.exists()) {
outputImage.delete();
}
outputImage.createNewFile();
} catch (IOException e) {
e.printStackTrace();
}
imageUri = Uri.fromFile(outputImage);
Intent intent = new Intent("android.intent.action.GET_CONTENT");
intent.setType("image/*");
intent.putExtra("crop", true);
intent.putExtra("scale", true);
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
startActivityForResult(intent, CROP_PHOTO);
}
});
最终效果:
(不知道为何,从相册中选出图片之后,存入手机存储的jpg文件是0kb大小,待解决)
第一行代码最后还提到了从拍照或者从相册中选出图片之后一般还要进行图片压缩,因为原图片的大小一般都比较大。图片压缩之前在做project的时候已经弄过,下次我做project回顾的时候,会把这一段讲一下。