本文主要介紹如何使用camera+surfaceview來自定義相機,我們使用自定義相機進行拍照、掃描二維碼等等。Camera是屬於硬件部分,在開發中會經常遇到如何使用相機。在上一篇文章中介紹了如何動態的申請權限,使用相機是需要動態申請權限的。
根據官方文檔介紹在拍攝照片時,用戶通常要先看到拍攝目標的預覽效果,然后再按下快門。為此,您可以使用 SurfaceView 繪制攝像頭傳感器需要獲取的內容的預覽效果。SurfaceView是由SurfaceHolder來操控。
一、首先創建一個SurfaceView,並實現SurfaceHolder的回調。由於Camera在SurfaceView中是通過SurfaceHolder 使得Surfaceview能夠預覽Camera返回的數據,因此我們需要實現SurfaceHolder 的回調,關於SurfaceView 、Surface 、 SurfaceHolder三者的關系可以參考這篇博客。
SurfaceHolder中有一個Callbcak接口,它有3個回調方法
-
surfaceCreated(SurfaceHolder holder)
surface第一次創建時回調 -
surfaceChanged(SurfaceHolder holder, int format, int width,
int height)
surface變化的時候回調(格式/大小) -
surfaceDestroyed(SurfaceHolder holder)
surface銷毀的時候回調
SurfaceView繼承自View,其中有兩個成員變量,一個是Surface對象,一個是SuraceHolder對象。
- SurfaceView把Surface顯示在屏幕上
- SurfaceView通過SuraceHolder告訴我們Surface的狀態(創建、變化、銷毀)
- 通過getHolder()方法獲得當前SurfaceView的SuraceHolder對象,然后就可以對SuraceHolder對象添加回調來監聽Surface的狀態
SurfaceView小結
- SurfaceView是一個view對象,用於在屏幕上顯示相機的預覽畫面
- SurfaceView中有兩個對象,Surface和SuraceHolder。 我們通過SuraceHolder中的回調可以知道Surface的狀態(創建、變化、銷毀)
- 通過getHolder()方法獲得當前SurfaceView的SuraceHolder對象
代碼如下:
1 package com.example.camera; 2 3 import androidx.annotation.NonNull; 4 import androidx.appcompat.app.AppCompatActivity; 5 import androidx.core.app.ActivityCompat; 6 import androidx.core.content.ContextCompat; 7 8 import android.Manifest; 9 import android.content.pm.PackageManager; 10 import android.hardware.Camera; 11 import android.os.Bundle; 12 import android.view.SurfaceHolder; 13 import android.view.SurfaceView; 14 import android.view.Window; 15 import android.view.WindowManager; 16 17 import java.io.IOException; 18 19 public class MainActivity extends AppCompatActivity { 20 private static final String TAG = "MainActivity"; 21 22 /* 23 * 創建SurfaceHolder,Camera,SurfaceViiew; 24 * */ 25 private SurfaceHolder surfaceHolder; 26 private Camera camera; 27 private SurfaceView surfaceView; 28 29 private static final String[] permission = new String[] { 30 Manifest.permission.CAMERA, 31 Manifest.permission.WRITE_EXTERNAL_STORAGE 32 }; 33 34 @Override 35 protected void onCreate(Bundle savedInstanceState) { 36 super.onCreate(savedInstanceState); 37 38 //去除頂部狀態欄 39 requestWindowFeature(Window.FEATURE_NO_TITLE); 40 getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,WindowManager.LayoutParams.FLAG_FULLSCREEN); 41 setContentView(R.layout.activity_main); 42 // Android 6.0相機動態權限檢查 43 if (ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA)== PackageManager.PERMISSION_GRANTED){ 44 }else { 45 ActivityCompat.requestPermissions(this,permission,1); 46 } 47 48 surfaceView = findViewById(R.id.surfaceView); 49 surfaceHolder = surfaceView.getHolder(); 50 surfaceHolder.addCallback(new SurfaceHolder.Callback() { 51 @Override 52 public void surfaceCreated(@NonNull SurfaceHolder holder) { 53 surfaceHolder = holder; 54 camera = Camera.open(Camera.CameraInfo.CAMERA_FACING_FRONT); 55 try { 56 camera.setPreviewDisplay(surfaceHolder); 57 } catch (IOException e) { 58 e.printStackTrace(); 59 } 60 camera.startPreview(); 61 62 } 63 64 @Override 65 public void surfaceChanged(@NonNull SurfaceHolder holder, int format, int width, int height) { 66 67 } 68 69 @Override 70 public void surfaceDestroyed(@NonNull SurfaceHolder holder) { 71 72 } 73 }); 74 75 76 77 78 79 } 80 }
activity_main.xml布局文件如下:
1 <?xml version="1.0" encoding="utf-8"?> 2 <FrameLayout xmlns:android="http://schemas.android.com/apk/res/android" 3 xmlns:app="http://schemas.android.com/apk/res-auto" 4 xmlns:tools="http://schemas.android.com/tools" 5 android:layout_width="match_parent" 6 android:layout_height="match_parent" 7 tools:context=".MainActivity"> 8 9 <SurfaceView 10 android:id="@+id/surfaceView" 11 android:layout_width="match_parent" 12 android:layout_height="match_parent" /> 13 <Button 14 android:id="@+id/switch_camer" 15 android:layout_width="wrap_content" 16 android:layout_height="wrap_content" 17 android:layout_gravity="bottom|center" 18 android:text="切換相機" 19 20 /> 21 22 </FrameLayout>
在這里我們已經初步的完成了在surfaceView中顯示camare的畫面了,但是打開之后,會發現顯示的畫面是扭曲的,無論怎么翻轉手機都是不可以的。這個問題后面解決。
在這里總結一下如何使用Camera+SurfaceView來顯示畫面:
1、首先獲得布局文件中定義的SurfaceView;
1 @Override 2 public void surfaceCreated(@NonNull SurfaceHolder holder) { 3 surfaceHolder = holder; 4 camera = Camera.open(1); 5 6 7 camera.setDisplayOrientation(90); 8 9 10 try { 11 camera.setPreviewDisplay(surfaceHolder); 12 } catch (IOException e) { 13 e.printStackTrace(); 14 } 15 camera.startPreview(); 16 17 }