Android Camera 攝像 demo

来源:http://www.cnblogs.com/xiaoxiaoqingyi/archive/2017/06/15/7016312.html
-Advertisement-
Play Games

google 在Android 5.0推出 Camera2 這個類,用於替換 Camera,但是Camera2要求android sdk 最低版本為 minSdkVersion = 21 (5.0系統),所以Camera2 還不能完全替換 Camera,在相容低版本的時候,還是需要兩者一起協同開發。 ...


      google 在Android 5.0推出 Camera2 這個類,用於替換 Camera,但是Camera2要求android sdk 最低版本為 minSdkVersion = 21 (5.0系統),所以Camera2 還不能完全替換 Camera,在相容低版本的時候,還是需要兩者一起協同開發。下麵我來說一下 Camera 的拍攝例子: 首先需要在xml 上佈局一個 SurfaceView 設置全屏  
<SurfaceView
android:id="@+id/surfaceView"
android:layout_width="match_parent"
android:layout_height="match_parent"/>

 

  同時也把狀態欄和titleBar隱藏了:
requestWindowFeature(Window.FEATURE_NO_TITLE);// 去掉標題欄
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);// 設置全屏

 

    然後獲取SurfacView 實例以及其持有者SurfaceHolder,接入SurfaceHolder.Callback回調,
mSurfaceView = (SurfaceView)findViewById(R.id.surfaceView);
mSurfaceHolder = mSurfaceView.getHolder();// 取得holder
mSurfaceHolder.addCallback(this); // holder加入回調介面
mSurfaceHolder.setKeepScreenOn(true);

 

  SurfaceHolder.Callback會在頁面Actvity 初始化完畢後調用,則在回調的surfaceChanged初始化Camera,也就是打開預覽頁面:  
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
if (mCamera != null) {
freeCameraResource();
}
 
try {
mCamera = Camera.open();
if (mCamera == null)
return;
mCamera.setDisplayOrientation(90);
mCamera.setPreviewDisplay(mSurfaceHolder);
parameters = mCamera.getParameters();// 獲得相機參數
 
List<Camera.Size> mSupportedPreviewSizes = parameters.getSupportedPreviewSizes();
List<Camera.Size> mSupportedVideoSizes = parameters.getSupportedVideoSizes();
optimalSize = CameraHelper.getOptimalVideoSize(mSupportedVideoSizes,
mSupportedPreviewSizes, height, width);
 
parameters.setPreviewSize(optimalSize.width, optimalSize.height); // 設置預覽圖像大小
 
parameters.set("orientation", "portrait");
List<String> focusModes = parameters.getSupportedFocusModes();
if (focusModes.contains("continuous-video")) {
parameters.setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_VIDEO);
}
mFpsRange = parameters.getSupportedPreviewFpsRange();
 
mCamera.setParameters(parameters);// 設置相機參數
mCamera.startPreview();// 開始預覽
 
 
}catch (Exception io){
io.printStackTrace();
}
}

 

  該方法返回了SurfaceView的寬與高,根據給出的尺寸與寬高比例,獲取一個最適配的預覽尺寸,你先看下麵有兩個參數:
 List<Camera.Size> mSupportedPreviewSizes = parameters.getSupportedPreviewSizes();
List<Camera.Size> mSupportedVideoSizes = parameters.getSupportedVideoSizes();

 

  這兩個隊列分別是 該相機支持的 預覽大小(一般就是拍照時照片的大小),另外一個就是支持適配的大小,因為都是隊列,說明相機支持很多組尺寸,而且,照片的尺寸與視頻的尺寸是不一樣的。我debug看了幾款手機,通常攝像支持的尺寸少一點,照片會多一些。這樣,我們就要通過剛剛方法給出的寬高,獲取一個最佳匹配的預覽尺寸:
public static Camera.Size getOptimalVideoSize(List<Camera.Size> supportedVideoSizes,
List<Camera.Size> previewSizes, int w, int h) {
// Use a very small tolerance because we want an exact match.
final double ASPECT_TOLERANCE = 0.1;
double targetRatio = (double) w / h;
 
// Supported video sizes list might be null, it means that we are allowed to use the preview
// sizes
List<Camera.Size> videoSizes;
if (supportedVideoSizes != null) {
videoSizes = supportedVideoSizes;
} else {
videoSizes = previewSizes;
}
Camera.Size optimalSize = null;
 
// Start with max value and refine as we iterate over available video sizes. This is the
// minimum difference between view and camera height.
double minDiff = Double.MAX_VALUE;
 
// Target view height
int targetHeight = h;
 
// Try to find a video size that matches aspect ratio and the target view size.
// Iterate over all available sizes and pick the largest size that can fit in the view and
// still maintain the aspect ratio.
for (Camera.Size size : videoSizes) {
double ratio = (double) size.width / size.height;
if (Math.abs(ratio - targetRatio) > ASPECT_TOLERANCE)
continue;
if (Math.abs(size.height - targetHeight) < minDiff && previewSizes.contains(size)) {
optimalSize = size;
minDiff = Math.abs(size.height - targetHeight);
}
}
 
// Cannot find video size that matches the aspect ratio, ignore the requirement
if (optimalSize == null) {
minDiff = Double.MAX_VALUE;
for (Camera.Size size : videoSizes) {
if (Math.abs(size.height - targetHeight) < minDiff && previewSizes.contains(size)) {
optimalSize = size;
minDiff = Math.abs(size.height - targetHeight);
}
}
}
return optimalSize;
}
  該方法是獲取最佳的預覽與攝像尺寸。然後設置預覽圖像大小:
parameters.setPreviewSize(optimalSize.width, optimalSize.height);
  可以通過獲取相機的參數實例,設置裡面各種效果,包括剛剛的預覽圖,前置攝像頭,閃光燈等。
parameters = mCamera.getParameters();// 獲得相機參數

 

設置好預覽已經相機參數,則打開:
mCamera.setParameters(parameters);// 設置相機參數
mCamera.startPreview();// 開始預覽

 

  那麼就進入一個預覽的拍攝頁面了,該頁面其實也可以用來做拍照。要想做拍攝,還要實例化MediaRecorder,然後傳入camera並初始化相應的參數:
mMediaRecorder.setCamera(mCamera);
mMediaRecorder.setOnErrorListener(this);
mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.DEFAULT );
mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);// 視頻源
 
// Use the same size for recording profile.
CamcorderProfile mProfile = CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH);
mProfile.videoFrameWidth = optimalSize.width;
mProfile.videoFrameHeight = optimalSize.height;
 
mMediaRecorder.setProfile(mProfile);
//該設置是為了抽取視頻的某些幀,真正錄視頻的時候,不要設置該參數
// mMediaRecorder.setCaptureRate(mFpsRange.get(0)[0]);//獲取最小的每一秒錄製的幀數
 
mMediaRecorder.setOutputFile(mVecordFile.getAbsolutePath());
 
mMediaRecorder.prepare();
mMediaRecorder.start();
    錄完的時候停止,需要重置才能再次使用。
try {
mMediaRecorder.stop();
mMediaRecorder.reset();
} catch (Exception e) {
e.printStackTrace();
}
 

 

當頁面destroy的時候,要記得釋放它們:
 private void releaseRecord() {
if (mMediaRecorder != null) {
mMediaRecorder.setPreviewDisplay(null);
mMediaRecorder.setOnErrorListener(null);
try {
mMediaRecorder.release();
} catch (IllegalStateException e) {
e.printStackTrace();
} catch (Exception e) {
e.printStackTrace();
}
}
mMediaRecorder = null;
}

 

  閃光燈關閉與開啟:
private void flashLightToggle(){
try {
if(isFlashLightOn){
parameters.setFlashMode(Camera.Parameters.FLASH_MODE_OFF);
mCamera.setParameters(parameters);
isFlashLightOn = false;
}else {
parameters.setFlashMode(Camera.Parameters.FLASH_MODE_TORCH);
mCamera.setParameters(parameters);
isFlashLightOn = true;
}
} catch (Exception e) {
e.printStackTrace();
}
}

 

  前後攝像頭切換,就要重新初始化 camera實例:
private void switchCamera(){
Camera.CameraInfo cameraInfo = new Camera.CameraInfo();
int cameraCount = Camera.getNumberOfCameras();//得到攝像頭的個數
 
for(int i = 0; i < cameraCount; i++ ) {
Camera.getCameraInfo(i, cameraInfo);//得到每一個攝像頭的信息
if(cameraPosition == 1) {
//現在是後置,變更為前置
if(cameraInfo.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {//代表攝像頭的方位,CAMERA_FACING_FRONT前置 CAMERA_FACING_BACK後置
mCamera.stopPreview();//停掉原來攝像頭的預覽
mCamera.release();//釋放資源
mCamera = null;//取消原來攝像頭
mCamera = Camera.open(i);//打開當前選中的攝像頭
try {
mCamera.setDisplayOrientation(90);
mCamera.setPreviewDisplay(mSurfaceHolder);//通過surfaceview顯示取景畫面
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
mCamera.setParameters(parameters);// 設置相機參數
mCamera.startPreview();//開始預覽
cameraPosition = 0;
break;
}
} else {
//現在是前置, 變更為後置
if(cameraInfo.facing == Camera.CameraInfo.CAMERA_FACING_BACK) {//代表攝像頭的方位,CAMERA_FACING_FRONT前置 CAMERA_FACING_BACK後置
mCamera.stopPreview();//停掉原來攝像頭的預覽
mCamera.release();//釋放資源
mCamera = null;//取消原來攝像頭
mCamera = Camera.open(i);//打開當前選中的攝像頭
try {
mCamera.setDisplayOrientation(90);
mCamera.setPreviewDisplay(mSurfaceHolder);//通過surfaceview顯示取景畫面
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
mCamera.setParameters(parameters);// 設置相機參數
mCamera.startPreview();//開始預覽
cameraPosition = 1;
break;
}
}
 
}
}

 

錄製時的頁面:   這就是使用 camera 來攝像的主要步驟,demo:https://github.com/xiaoxiaoqingyi/android-CameraVideo   如果你想瞭解 Camera2,你也可以看看google 的 Camera2 官方例子: https://github.com/googlesamples/android-Camera2Basic  
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • KVO
    1.KVO概念 KVO即鍵值觀察,它提供一種機制,當被觀察的對象的屬性發生改變後,對象會接收到通知,從而做出相應的改變。 2.KVO實現原理 這裡要說一個isa指針,在Objective-C中,任何類的定義都是對象。類和類的實例(對象)沒有任何本質上的區別。任何對象都有isa指針。 那麼什麼是類呢? ...
  • 最近做項目要求使用到網路,想來想去選擇了AsyncHttpClient框架開進行APP開發。在這裡把我工作期間遇到的問題以及對AsyncHttpClient的使用經驗做出相應總結,希望能對您的學習有所幫助。 首先按照慣例先來簡單瞭解一些AsyncHttpClient網路框架的一些知識。 1、簡介 A ...
  • description 在開發過程中, 往往會有很多的model來裝載屬性. 而在開發期間經常會進行調試查看model里的屬性值是否正確. 那麼問題來了, 在objective-c里使用NSLog("%@",model)這行代碼列印出來的卻是model的地址. 不是我們所想要的結果~! 看圖: 那麼 ...
  • 登錄的業務邏輯{ http:是短連接. 伺服器如何判斷當前用戶是否登錄? // 1. 如果是即時通信類:長連接. // 如何保證伺服器跟客戶端保持長連接狀態? // "心跳包" 用來檢測用戶是否線上!用來做長連接! http:短連接使用token 機制來驗證用戶安全性 // token 值: 登錄令 ...
  • 有些時候我們使用Service的時需要採用隱私啟動的方式,但是Android 5.0一齣來後,其中有個特性就是Service Intent must be explitict,也就是說從Lollipop開始,service服務必須採用顯示方式啟動。 而android源碼是這樣寫的(源碼位置:sdk/ ...
  • 轉自 http://www.jianshu.com/p/6588c69b42cf ...
  • 基於AS的採用Kotlin語言開發的動畫漸入的弧形菜單(附java代碼),具體看效果... ...
  • NSCharacterSet 對於string處理,在IOS生態系統中經常用到且用錯的重要組成部分:NSCharacterSet。下麵詳解一下: NSCharacterSet ,以及它的可變版本NSMutableCharacterSet,用面向對象的方式來表示一組Unicode字元。它經常與NSSt ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...