Android實(shí)現(xiàn)調(diào)用攝像頭和相冊(cè)的方法
Android調(diào)用攝像頭是很方便的。先看一下界面

布局文件activity_main.xml源碼
<?xml version="1.0" encoding="utf-8"?>
<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:id="@+id/take_photo"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="啟動(dòng)相機(jī)" />
<Button
android:id="@+id/choose_from_album"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="從相冊(cè)中選擇圖片" />
<ImageView
android:id="@+id/picture"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_horizontal" />
</LinearLayout>
因?yàn)樯婕暗较騍D卡寫(xiě)入數(shù)據(jù),所有需要在AndroidMainfest.xml中聲明響應(yīng)權(quán)限
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
MainActivity.java源碼
package com.example.luoxn28.activity;
import android.annotation.TargetApi;
import android.content.ContentUris;
import android.content.Intent;
import android.database.Cursor;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.net.Uri;
import android.os.Bundle;
import android.os.Environment;
import android.provider.DocumentsContract;
import android.provider.MediaStore;
import android.support.v7.app.ActionBarActivity;
import android.view.View;
import android.widget.Button;
import android.widget.ImageView;
import android.widget.Toast;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.IOException;
public class MainActivity extends ActionBarActivity {
private static final String TAG = "hdu";
public static final int TAKE_PHOTO = 1;
public static final int CROP_PHOTO = 2;
public static final int CHOOSE_PHOTO = 3;
private Button takePhoto;
private ImageView picture;
private Uri imageUri;
private Button chooseFromAlbum;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//requestWindowFeature(Window.FEATURE_NO_TITLE);
setContentView(R.layout.activity_main);
takePhoto = (Button) findViewById(R.id.take_photo);
picture = (ImageView) findViewById(R.id.picture);
chooseFromAlbum = (Button) findViewById(R.id.choose_from_album);
takePhoto.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// 創(chuàng)建File對(duì)象,用于存儲(chǔ)拍攝后照片
File saveImage = new File(Environment.getExternalStorageDirectory(), "saveImage.jpg");
try {
if (saveImage.exists()) {
saveImage.delete();
}
saveImage.createNewFile();
}
catch (IOException ex) {
ex.printStackTrace();
}
imageUri = Uri.fromFile(saveImage);
Intent intent = new Intent("android.media.action.IMAGE_CAPTURE");
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
// 啟動(dòng)相機(jī)
startActivityForResult(intent, TAKE_PHOTO);
}
});
chooseFromAlbum.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
Intent intent = new Intent("android.intent.action.GET_CONTENT");
intent.setType("image/*");
// 打開(kāi)相冊(cè)
startActivityForResult(intent, CHOOSE_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");
intent.setDataAndType(imageUri, "image/*");
intent.putExtra("scale", true);
intent.putExtra(MediaStore.EXTRA_OUTPUT, imageUri);
// 啟動(dòng)裁剪程序
startActivityForResult(intent, CROP_PHOTO);
}
break;
case CROP_PHOTO:
if (resultCode == RESULT_OK) {
try {
Bitmap bitmap = BitmapFactory.decodeStream(getContentResolver().openInputStream(imageUri));
// 顯示裁剪后的圖片
picture.setImageBitmap(bitmap);
}
catch (FileNotFoundException ex) {
ex.printStackTrace();
}
}
break;
case CHOOSE_PHOTO:
if (resultCode == RESULT_OK) {
handleImage(data);
}
break;
default:
break;
}
}
// 只在Android4.4及以上版本使用
@TargetApi(19)
private void handleImage(Intent data) {
String imagePath = null;
Uri uri = data.getData();
if (DocumentsContract.isDocumentUri(this, uri)) {
// 通過(guò)document id來(lái)處理
String docId = DocumentsContract.getDocumentId(uri);
if ("com.android.providers.media.documents".equals(uri.getAuthority())) {
// 解析出數(shù)字id
String id = docId.split(":")[1];
String selection = MediaStore.Images.Media._ID + "=" + id;
imagePath = getImagePath(MediaStore.Images.Media.EXTERNAL_CONTENT_URI, selection);
}
else if ("com.android.providers.downloads.documents".equals(uri.getAuthority())) {
Uri contentUri = ContentUris.withAppendedId(Uri.parse("content://downloads/public_downloads"),
Long.valueOf(docId));
imagePath = getImagePath(contentUri, null);
}
}
else if ("content".equals(uri.getScheme())) {
// 如果不是document類(lèi)型的Uri,則使用普通方式處理
imagePath = getImagePath(uri, null);
}
// 根據(jù)圖片路徑顯示圖片
displayImage(imagePath);
}
private String getImagePath(Uri uri, String selection) {
String path = null;
// 通過(guò)Uri和selection來(lái)獲取真實(shí)圖片路徑
Cursor cursor = getContentResolver().query(uri, null, selection, null, null);
if (cursor != null) {
if (cursor.moveToFirst()) {
path = cursor.getString(cursor.getColumnIndex(MediaStore.Images.Media.DATA));
}
cursor.close();
}
return path;
}
private void displayImage(String imagePath) {
if (imagePath != null) {
Bitmap bitmap = BitmapFactory.decodeFile(imagePath);
picture.setImageBitmap(bitmap);
}
else {
Toast.makeText(this, "failed to get image", Toast.LENGTH_SHORT).show();
}
}
}
調(diào)用攝像頭拍照
在MainActivity 中要做的第一件事自然是分別獲取到 Button 和 ImageView 的實(shí)例,并給 Button 注冊(cè)上點(diǎn)擊事件,然后在 Button的點(diǎn)擊事件里開(kāi)始處理調(diào)用攝像頭的邏輯,我們重點(diǎn)看下這部分代碼。
首先這里創(chuàng)建了一個(gè) File 對(duì)象,用于存儲(chǔ)攝像頭拍下的圖片,這里我們把圖片命名為saveImage.jpg ,并將它存放在手機(jī)SD卡的根目錄下,調(diào) 用 Environment 的getExternalStorageDirectory()方法獲取到的就是手機(jī) SD 卡的根目錄。然后再調(diào)用 Uri 的fromFile()方法將 File 對(duì)象轉(zhuǎn)換成 Uri 對(duì)象,這個(gè) Uri 對(duì)象標(biāo)識(shí)著 saveImage.jpg 這張圖片的唯一地址。 接著構(gòu)建出一個(gè) Intent對(duì)象, 并將這個(gè) Intent的 action指定為android.media.action.IMAGE_CAPTURE,再調(diào)用 Intent 的 putExtra()方法指定圖片的輸出地址,這里填入剛剛得到的 Uri 對(duì)象,最后調(diào)用 startActivityForResult()來(lái)啟動(dòng)活動(dòng)。由于我們使用的是一個(gè)隱式Intent,系統(tǒng)會(huì)找出能夠響應(yīng)這個(gè) Intent 的活動(dòng)去啟動(dòng),這樣照相機(jī)程序就會(huì)被打開(kāi),拍下的照片將會(huì)輸出到 saveImage.jpg 中。
注意剛才我們是使用 startActivityForResult()來(lái)啟動(dòng)活動(dòng)的,因此拍完照后會(huì)有結(jié)果返回到 onActivityResult()方法中。如果發(fā)現(xiàn)拍照成功,則會(huì)再次構(gòu)建出一個(gè) Intent 對(duì)象,并把它的 action 指定為 com.android.camera.action.CROP。這個(gè) Intent 是用于對(duì)拍出的照片進(jìn)行裁剪注意剛才我們是使用 startActivityForResult()來(lái)啟動(dòng)活動(dòng)的,因此拍完照后會(huì)有結(jié)果返回到 onActivityResult()方法中。如果發(fā)現(xiàn)拍照成功,則會(huì)再次構(gòu)建出一個(gè) Intent 對(duì)象,并把它的 action 指定為 com.android.camera.action.CROP。這個(gè) Intent 是用于對(duì)拍出的照片進(jìn)行裁剪
從相冊(cè)中選擇照片
在 "從相冊(cè)中選擇圖片"按鈕的點(diǎn)擊事件里我們同樣創(chuàng)建了一個(gè) File 對(duì)象,用于存儲(chǔ)從相冊(cè)中選擇的圖片。然后構(gòu)建出一個(gè) Intent 對(duì)象,并將它的 action 指定為android.intent.action.GET_CONTENT。接著給這個(gè) Intent 對(duì)象設(shè)置一些必要的參數(shù),包括是否允許縮放和裁剪、圖片的輸出位置等。最后調(diào)用 startActivityForResult()方法,就可以打開(kāi)相冊(cè)程序選擇照片了。
注意在調(diào)用 startActivityForResult()方法的時(shí)候,我們給第二個(gè)參數(shù)傳入的值仍然是CROP_PHOTO 常量,這樣的好處就是從相冊(cè)選擇好照片之后,會(huì)直接進(jìn)入到 CROP_PHOTO的 case 下將圖片顯示出來(lái), 這樣就可以復(fù)用之前寫(xiě)好的顯示圖片的邏輯, 不用再編寫(xiě)一遍了。
參考資料
1、《第一行代碼-Android》調(diào)用攝像頭章節(jié)
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android實(shí)現(xiàn)手機(jī)攝像頭的自動(dòng)對(duì)焦
- android開(kāi)發(fā)之調(diào)用手機(jī)的攝像頭使用MediaRecorder錄像并播放
- Android開(kāi)發(fā)教程之調(diào)用攝像頭功能的方法詳解
- Android實(shí)現(xiàn)調(diào)用攝像頭進(jìn)行拍照功能
- Android實(shí)現(xiàn)調(diào)用攝像頭拍照與視頻功能
- Android調(diào)用前后攝像頭同時(shí)工作實(shí)例代碼
- Android實(shí)現(xiàn)調(diào)用攝像頭
- Android調(diào)用系統(tǒng)攝像頭拍照并顯示在ImageView上
- Android讀取本地圖庫(kù)與調(diào)用攝像頭拍攝
- Android調(diào)用手機(jī)攝像頭的方法
相關(guān)文章
Android入門(mén)之Activity間互相傳值詳解
我們?cè)谥暗腟ervice篇章中看到了一種putExtras和getExtras來(lái)進(jìn)行activity與service間的傳值。而恰恰這種傳值其實(shí)也是Android里的通用傳值法。它同樣可以適用在activity與activity間傳值,本文就來(lái)和大家詳細(xì)講講2022-12-12
android真機(jī)調(diào)試時(shí)無(wú)法顯示logcat信息的解決方法介紹
以下是對(duì)android真機(jī)調(diào)試時(shí)無(wú)法顯示logcat信息的解決方法進(jìn)行了詳細(xì)的分析介紹,需要的朋友可以過(guò)來(lái)參考下2013-07-07
Android編程實(shí)現(xiàn)播放視頻時(shí)切換全屏并隱藏狀態(tài)欄的方法
這篇文章主要介紹了Android編程實(shí)現(xiàn)播放視頻時(shí)切換全屏并隱藏狀態(tài)欄的方法,結(jié)合實(shí)例形式分析了Android視頻播放事件響應(yīng)及相關(guān)屬性設(shè)置操作技巧,需要的朋友可以參考下2017-08-08
Android基礎(chǔ)開(kāi)發(fā)小案例之短信發(fā)送器
這篇文章主要為大家詳細(xì)介紹了Android基礎(chǔ)開(kāi)發(fā)小案例之短信發(fā)送器的具體實(shí)現(xiàn)代碼,感興趣的小伙伴們可以參考一下2016-05-05
Android TextView設(shè)置不同的顏色字體
這篇文章主要為大家詳細(xì)介紹了Android TextView設(shè)置不同的顏色字體,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-12-12
Android啟動(dòng)頁(yè)面定時(shí)跳轉(zhuǎn)的三種方法
這篇文章主要介紹了Android啟動(dòng)頁(yè)面定時(shí)跳轉(zhuǎn)的三種方法,實(shí)現(xiàn)打開(kāi)一個(gè)Android手機(jī)APP的歡迎界面后跳轉(zhuǎn)到指定界面的效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11
代碼分析Android實(shí)現(xiàn)側(cè)滑菜單
現(xiàn)在app越來(lái)越注重用戶(hù)體驗(yàn),本文給大家分析android實(shí)現(xiàn)側(cè)滑菜單的代碼,代碼簡(jiǎn)單易懂,感興趣的朋友一起看看吧2015-11-11

