獲取Android設備上的詳細的攝像頭信息

jopen 10年前發布 | 53K 次閱讀 Android 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();   
            }   
        }   
    }   
}

 

 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!