獲取Android設備上的詳細的攝像頭信息
如何獲取Android設備上的詳細的攝像頭信息呢? 目前Samsung的Galaxy Tab和Nexus S均有前置攝像頭,獲取Android攝像頭的詳細信息,在Android 2.3SDK中得到了增強:
在android.hardware.Camera類中,API Level 9的SDK中加入了兩個比較重要的方法,使用getNumberOfCameras這個static類型方法可以獲取當前Android設備上的攝像頭數量,比如Nexus S有兩個,方法原型如下
public static int getNumberOfCameras ()
而對于具體的每個攝像頭的信息,可以通過Camera類的getCameraInfo()這個靜態方法獲取,該方法有兩個參數,參數一的ID,我們通過 getNumberOfCameras獲取的值減1即可,類似數組索引從0開始一樣,用循環遍歷每個攝像頭信息,參數二是 android.hardware.Camera.CameraInfo類,有關getCameraInfo方法的原型如下:
public static void getCameraInfo (int cameraId, Camera.CameraInfo cameraInfo)
對于Camera.CameraInfo類而言,比較簡單,包含兩個字段
public int facing 代表攝像頭的方位,目前有定義值兩個分別為CAMERA_FACING_FRONT前置和CAMERA_FACING_BACK后置
public int orientation 下面是拍照的旋轉方向,一般自然些有0度、90度、180度和270度,這樣可以獲取我們正確的手握設備是橫著還是豎著,有關拍照時的方向設置,可以參考下面的代碼設置
public static void setCameraDisplayOrientation(Activity activity, int cameraId, android.hardware.Camera camera) { android.hardware.Camera.CameraInfo info = new android.hardware.Camera.CameraInfo(); android.hardware.Camera.getCameraInfo(cameraId, info); int rotation = activity.getWindowManager().getDefaultDisplay() .getRotation(); int degrees = 0; switch (rotation) { case Surface.ROTATION_0: degrees = 0; break; case Surface.ROTATION_90: degrees = 90; break; case Surface.ROTATION_180: degrees = 180; break; case Surface.ROTATION_270: degrees = 270; break; } int result; if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) { result = (info.orientation + degrees) % 360; result = (360 - result) % 360; // compensate the mirror } else { // back-facing result = (info.orientation - degrees + 360) % 360; } camera.setDisplayOrientation(result); }
使用攝像頭拍照
1. 使用 SurfaceView 控件來顯示攝像頭捕捉到的畫面
<SurfaceView android:layout_width="fill_parent" android:layout_height="240dip" android:id="@+id/surfaceView" />
2. 具體細節
/* 獲取 SurfaceView 控件 */ SurfaceView surfaceView = (SurfaceView)this.findViewById(R.id.surfaceView); /* 設置分辨率 */ surfaceView.getHolder().setFixedSize(176, 144); /*下面設置Surface不維護自己的緩沖區,而是等待屏幕的渲染引擎將內容推送到用戶面前*/ surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); /* 打開攝像頭,注意這里是 android.hardware.Camera */ Camera camera = Camera.open(); /* 為 Camera 設置攝像參數 */ Camera.Parameters parameters = camera.getParameters(); /* 設置預覽照片的大小,此處設置為全屏 */ WindowManager wm = (WindowManager) getSystemService(Context.WINDOW_SERVICE); // 獲取當前屏幕管理器對象 Display display = wm.getDefaultDisplay(); // 獲取屏幕信息的描述類 parameters.setPreviewSize(display.getWidth(), display.getHeight()); // 設置 /* 每秒從攝像頭捕獲5幀畫面, */ parameters.setPreviewFrameRate(5); /* 設置照片的輸出格式:jpg */ parameters.setPictureFormat(PixelFormat.JPEG); /* 照片質量 */ parameters.set("jpeg-quality", 85); /* 設置照片的大小:此處照片大小等于屏幕大小 */ parameters.setPictureSize(display.getWidth(), display.getHeight()); /* 將參數對象賦予到 camera 對象上 */ camera.setParameters(parameters); /* 設置用 SurfaceView 作為承載鏡頭取景畫面的顯示 */ camera.setPreviewDisplay(surfaceView.getHolder()); /* 開始預覽 */ camera.startPreview(); /* 自動對焦 */ camera.autoFocus(null); /* 拍照片 */ camera.takePicture(null, null, null, jpegCallback); /* 停止預覽 */ camera.stopPreview(); /* 釋放攝像頭 */ camera.release();
3. 添加使用攝像頭的權限
<uses-permission android:name="android.permission.CAMERA"/>
4. 測試: 目前模擬器不支持拍照環境的模擬,必須使用真實手機測試。
5. 代碼清單
** string values : strings.xml
<?xml version="1.0" encoding="utf-8"?> <resources> <string name="app_name">手機拍照程序</string> </resources> <?xml version="1.0" encoding="utf-8"?> <resources> <string name="app_name">手機拍照程序</string> </resources>
** AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="wjh.android.takepicture" android:versionCode="1" android:versionName="1.0"> <application android:icon="@drawable/icon" android:label="@string/app_name"> <!-- android:screenOrientation="landscape" 表示橫向界面 --> <activity android:name=".MainActicity" android:label="@string/app_name" android:screenOrientation="landscape"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> <uses-sdk android:minSdkVersion="8" /> <!-- 在SDCard中創建與刪除文件權限 --> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> <!-- 往SDCard寫入數據權限 --> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <!-- 申請使用攝像頭的權限 --> <uses-permission android:name="android.permission.CAMERA"/> </manifest> <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="wjh.android.takepicture" android:versionCode="1" android:versionName="1.0"> <application android:icon="@drawable/icon" android:label="@string/app_name"> <!-- android:screenOrientation="landscape" 表示橫向界面 --> <activity android:name=".MainActicity" android:label="@string/app_name" android:screenOrientation="landscape"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> </application> <uses-sdk android:minSdkVersion="8" /> <!-- 在SDCard中創建與刪除文件權限 --> <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> <!-- 往SDCard寫入數據權限 --> <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/> <!-- 申請使用攝像頭的權限 --> <uses-permission android:name="android.permission.CAMERA"/> </manifest> ** main.xml view plaincopy to clipboardprint? <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <SurfaceView android:id="@+id/surfaceView" android:layout_width="fill_parent" android:layout_height="fill_parent" /> </LinearLayout> <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:orientation="vertical" android:layout_width="fill_parent" android:layout_height="fill_parent" > <SurfaceView android:id="@+id/surfaceView" android:layout_width="fill_parent" android:layout_height="fill_parent" /> </LinearLayout>
** MainActicity
public class MainActicity extends Activity { private Camera camera; private boolean preview = false ; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); /* *設置窗口屬性:一定要在 setContentView(R.layout.main) 之前 */ // 窗口標題 requestWindowFeature(Window.FEATURE_NO_TITLE); // 全屏 getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN); setContentView(R.layout.main); SurfaceView surfaceView = (SurfaceView) findViewById(R.id.surfaceView); surfaceView.getHolder().setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); surfaceView.getHolder().setFixedSize(176, 164); surfaceView.getHolder().addCallback(new SurfaceViewCallback()); } private final class SurfaceViewCallback implements Callback { /** * surfaceView 被創建成功后調用此方法 */ @Override public void surfaceCreated(SurfaceHolder holder) { /* * 在SurfaceView創建好之后 打開攝像頭 * 注意是 android.hardware.Camera */ camera = Camera.open(); Camera.Parameters parameters = camera.getParameters(); /* 設置預覽照片的大小,此處設置為全屏 */ WindowManager wm = (WindowManager) getSystemService(Context.WINDOW_SERVICE); // 獲取當前屏幕管理器對象 Display display = wm.getDefaultDisplay(); // 獲取屏幕信息的描述類 parameters.setPreviewSize(display.getWidth(), display.getHeight()); // 設置 /* 每秒從攝像頭捕獲5幀畫面, */ parameters.setPreviewFrameRate(5); /* 設置照片的輸出格式:jpg */ parameters.setPictureFormat(PixelFormat.JPEG); /* 照片質量 */ parameters.set("jpeg-quality", 85); /* 設置照片的大小:此處照片大小等于屏幕大小 */ parameters.setPictureSize(display.getWidth(), display.getHeight()); /* 將參數對象賦予到 camera 對象上 */ camera.setParameters(parameters); preview = true; } @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) { } /** * SurfaceView 被銷毀時釋放掉 攝像頭 */ @Override public void surfaceDestroyed(SurfaceHolder holder) { if(camera != null) { /* 若攝像頭正在工作,先停止它 */ if(preview) { camera.stopPreview(); preview = false; } camera.release(); } } } /** * 手機鍵盤按鍵事件 * 返回 true, 將阻止事件繼續傳遞,例如搜索鍵,他默認會觸發和打開系統的搜索引擎。返回true后,將不會觸發。 */ @Override public boolean onKeyDown(int keyCode, KeyEvent event) { /* * event.getRepeatCount() 為重復按鍵的次數,例如,快速地對某個鍵連續按了兩次,則此值為一,表示重復了一次。往上可以累推。 * 按鍵只被按了一次,則此值為 0。 * 這有點類似于鼠標的 "單擊" 和 "雙擊"。 */ if(camera != null && event.getRepeatCount() == 0 ) { switch (keyCode) { case KeyEvent.KEYCODE_SEARCH: // 搜索鍵 /* 按下搜索鍵自動對焦 , 如果要關注它的事件, * 可以實現 AutoFocusCallback 接口,并實例化其對象傳入 */ camera.autoFocus(null); break; case KeyEvent.KEYCODE_CAMERA: // 拍照鍵 case KeyEvent.KEYCODE_DPAD_CENTER: // 中間確認鍵 /* * @param shutter : 照片被捕獲之后的回調對象 * @param raw : 此回調對象可以生產為壓縮的圖片數據 * @param jpeg : 此回調對象可以產生壓縮后的圖片數據,其onPictureTaken將被調用 */ camera.takePicture(null, null, new TakePictureCallback()); /* 拍完照后回到預覽狀態,繼續取景 -- 錯誤的方式 */ // camera.startPreview();必須寫在 onPictureTaken 方法內部,因為 takePicture 內部是另開了一條線程異步的完成保存照片等操作。 // 雖然 takePicture 方法完成了,但是并不代表其內部的工作全部完成,也不代表攝像頭以及從上一次“拍照”任務中工作完畢 break; default: break; } return true; } return super.onKeyDown(keyCode, event); } /** * 處理照片被拍攝之后的事件 */ private final class TakePictureCallback implements PictureCallback { @Override public void onPictureTaken(byte[] data, Camera camera) { Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length); /* 照片將被保存到 SD 卡跟目錄下,文件名為系統時間,后綴名為".jpg" */ File file = new File(Environment.getExternalStorageState(), System.currentTimeMillis() + ".jpg"); try { FileOutputStream fos = new FileOutputStream(file); /* 位圖格式為JPEG * 參數二位 0-100 的數值,100為最大值,表示無損壓縮 * 參數三傳入一個輸出流對象,將圖片數據輸出到流中 */ bitmap.compress(CompressFormat.JPEG, 100, fos); fos.close(); /* 拍完照后回到預覽狀態,繼續取景 */ camera.startPreview(); } catch (IOException e) { e.printStackTrace(); } } } }