Android SurfaceView 自定義相機

  時間過的真快,轉眼就要到五一假期,在這裏提前祝大家五一快樂;今天有時間,簡單的寫點最近項目中遇到的坑。具體代碼我會貼出來,我就不放在GitHub上了,有感興趣的可以去

我的博客地址
www.softwareboy.com.cn


雖然很醜,但是實用就可以了,言歸正傳!;

一.SurfaceView API介紹
  SurfaceView是視圖(View)的繼承類,這個視圖裏內嵌了一個專門用於繪製的Surface。你可以控制這個Surface的格式和尺寸。Surfaceview控制這個Surface的繪製位置。
  Surface是縱深排序(Z-ordered)的,這表明它總在自己所在窗口的後面。surfaceview提供了一個可見區域,只有在這個可見區域內 的surface部分內容纔可見,可見區域外的部分不可見。surface的排版顯示受到視圖層級關係的影響,它的兄弟視圖結點會在頂端顯示。這意味者 surface的內容會被它的兄弟視圖遮擋,這一特性可以用來放置遮蓋物(overlays)(例如,文本和按鈕等控件)。注意,如果surface上面 有透明控件,那麼它的每次變化都會引起框架重新計算它和頂層控件的透明效果,這會影響性能。
  你可以通過SurfaceHolder接口訪問這個surface,getHolder()方法可以得到這個接口。
  surfaceview變得可見時,surface被創建;surfaceview隱藏前,surface被銷燬。這樣能節省資源。如果你要查看 surface被創建和銷燬的時機,可以重載surfaceCreated(SurfaceHolder)和 surfaceDestroyed(SurfaceHolder)。
  surfaceview的核心在於提供了兩個線程:UI線程和渲染線程。這裏應注意:
  1> 所有SurfaceView和SurfaceHolder.Callback的方法都應該在UI線程裏調用,一般來說就是應用程序主線程。渲染線程所要訪問的各種變量應該作同步處理。
  2> 由於surface可能被銷燬,它只在SurfaceHolder.Callback.surfaceCreated()和SurfaceHolder.Callback.surfaceDestroyed()之間有效,所以要確保渲染線程訪問的是合法有效的surface。
二.Android Camera
  通過以下兩種方式,可以在應用程序中使用攝像機
     1.使用現有應用程序中Android攝像頭應用程序
     2.直接使用應用程序提供的Android攝像頭API

   使用現有應用程序的Android攝像頭應用程序

使用 MediaStore.ACTION_IMAGE_CAPTURE 啓動安裝在手機上的攝像頭應用程序。它的語法下面給出:
Intent intent = new Intent(android.provider.MediaStore.ACTION_IMAGE_CAPTURE);
startActivityForResult(intent,0)
其實就這樣,然後在Activity中調onActivityResult方法就可以針對照片進行相應處理了;
*這樣的處理方式並不能滿足當今人類的需求啊,例如相機我們要進行美顏呀,什麼添加水印啊等等動作;這就不能去使用手機自帶的相機了,還是自己寫一個吧,感覺還蠻好用的;廢話不多說了,先粘貼出來我的佈局;這個我就不講解了!

佈局文件的效果圖

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <SurfaceView
        android:id="@+id/id_graph_sfv"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
    <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="150dp"
        android:layout_gravity="bottom"
        android:background="#393A3F"
        android:gravity="center_vertical"
        android:orientation="horizontal"
        android:padding="20dp">
        <ImageButton
            android:id="@+id/id_reset_camera"
            android:layout_width="50dp"
            android:layout_height="50dp"
            android:layout_centerVertical="true"
            android:background="@drawable/reset"
            android:visibility="visible" />
        <ImageButton
            android:id="@+id/reset"
            android:layout_width="50dp"
            android:layout_height="50dp"
            android:layout_centerVertical="true"
            android:background="@drawable/pf"
            android:visibility="gone" />
        <ImageButton
            android:id="@+id/id_ok"
            android:layout_width="50dp"
            android:layout_height="50dp"
            android:layout_alignParentRight="true"
            android:layout_centerVertical="true"
            android:background="@drawable/ok" />
        <FrameLayout
            android:id="@+id/frameLayout"
            android:layout_width="80dp"
            android:layout_height="80dp"
            android:layout_centerInParent="true"
            android:background="@drawable/stop_bc"
            android:padding="3dp"
            android:visibility="visible">
            <ImageButton
                android:id="@+id/imageButton"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:background="@drawable/camera" />
        </FrameLayout>
    </RelativeLayout>
</FrameLayout>
  所需要的圖片都是自己找的,所以大家上網發掘下就出來了.
import android.app.Activity;
import android.content.DialogInterface;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.ImageFormat;
import android.graphics.Matrix;
import android.hardware.Camera;
import android.os.Bundle;
import android.os.Environment;
import android.support.annotation.Nullable;
import android.support.v7.app.AlertDialog;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.View;
import android.view.Window;
import android.widget.ImageButton;

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
/**
* 在這裏需要注意的是,相機的釋放一定要處理,在針對Android6.0 和7.0系統的*時候,我們的動態權限申請。這裏我就不寫出來了。GitHub上一堆可用方法;
**/
public class Photograph extends Activity implements SurfaceHolder.Callback {

    private Camera mCamera;//聲明相機(不要倒錯包)
    private SurfaceView mPrevice;//聲明視圖
    private SurfaceHolder mHolder;//
    private ImageButton id_graph_camera;//
    private ImageButton id_ok;//
    private ImageButton reset;//
    private ImageButton id_reset_camera;//
    private Bitmap bitmap = null;//
    private String picUrl;//
    private int cameraPosition = 1; // 0代表前置攝像頭,1代表後置攝像頭
    private Bitmap bm = null;//
    private Camera.Parameters parameters;//

    //回調拍照聲音的,想要靜音拍照的同學,你們懂的!切記不要做壞事兒哦
    private Camera.ShutterCallback shutterCallback = new Camera.ShutterCallback() {
        @Override
        public void onShutter() {

        }
    };

    //相機拍照後對照片的回調
    private Camera.PictureCallback mPictureCallback = new Camera.PictureCallback() {
        @Override
        public void onPictureTaken(byte[] data, Camera camera) {
            File tempFile = new File("/sdcard/" + System.currentTimeMillis() + ".png");
            try {
                FileOutputStream fos = new FileOutputStream(tempFile);
                fos.write(data);
                fos.close();

                bm = BitmapFactory.decodeByteArray(data, 0, data.length);

                Matrix matrix = new Matrix();
                if (cameraPosition == 1) {
                    matrix.postRotate(90);
                } else {
                    matrix.postRotate(270);
                }

                bitmap = Bitmap.createBitmap(bm, 0, 0, bm.getWidth(), bm.getHeight(), matrix, true);
                picUrl = saveImage(bitmap).getAbsolutePath();

                id_reset_camera.setVisibility(View.VISIBLE);
                reset.setVisibility(View.GONE);
                id_ok.setVisibility(View.VISIBLE);

            } catch (FileNotFoundException e) {
                e.printStackTrace();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    };

    @Override
    protected void onCreate(@Nullable Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        setContentView(R.layout.photograph);
        mPrevice = (SurfaceView) findViewById(R.id.id_graph_sfv);
        id_graph_camera = (ImageButton) findViewById(R.id.imageButton);
        id_ok = (ImageButton) findViewById(R.id.id_ok);
        reset = (ImageButton) findViewById(R.id.reset);
        id_reset_camera = (ImageButton) findViewById(R.id.id_reset_camera);
        id_reset_camera.setVisibility(View.GONE);
        reset.setVisibility(View.VISIBLE);
        id_ok.setVisibility(View.GONE);
        mHolder = mPrevice.getHolder();
        mHolder.addCallback(this);
        //判斷內存卡是否存在的
        checkSoftStage();
        //重新啓動相機,
        id_reset_camera.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                mCamera.startPreview();
                bm.recycle();
                bitmap.recycle();
                reset.setVisibility(View.VISIBLE);
                id_ok.setVisibility(View.GONE);
                id_reset_camera.setVisibility(View.GONE);
            }
        });
        //切換攝像頭
        reset.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //切換前後攝像頭
                int cameraCount = 0;
                Camera.CameraInfo cameraInfo = new Camera.CameraInfo();
                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.setPreviewDisplay(mHolder);//通過surfaceview顯示取景畫面
                            } catch (IOException e) {
                                e.printStackTrace();
                            }
                            setStartPrevicw(mCamera, mHolder);
                            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.setPreviewDisplay(mHolder);//通過surfaceview顯示取景畫面
                            } catch (IOException e) {
                                e.printStackTrace();
                            }
                            setStartPrevicw(mCamera, mHolder);
                            cameraPosition = 1;
                            break;
                        }
                    }
                }
            }
        });
        //確定了是你要的照片,你可以選擇保存在相冊,也可以選擇跳轉到你想要顯示的頁面中,我這邊選擇在下個頁面顯示
        id_ok.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(Photograph.this, MessageSendActivity.class);
                intent.putExtra("TYPE", StaticInApp.CAMERA_IMAGE);
                StaticInApp.image_cache.add(picUrl);
                startActivity(intent);
                Photograph.this.finish();
            }
        });
        //點擊開始拍照
        id_graph_camera.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                startCamera();
            }
        });
    }
    /**
     * 檢測手機是否存在SD卡,網絡連接是否打開
     */
    private void checkSoftStage() {
        if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {  //判斷是否存在SD卡
            String rootPath = Environment.getExternalStorageDirectory().getPath();  //獲取SD卡的根目錄
            File file = new File(rootPath);
            if (!file.exists()) {
                file.mkdir();
            }
        } else {
            new AlertDialog.Builder(this).setMessage("檢測到手機沒有存儲卡!請插入手機存儲卡再開啓本應用。")
                    .setPositiveButton("確定", new DialogInterface.OnClickListener() {
                        @Override
                        public void onClick(DialogInterface dialog, int which) {
                            finish();
                        }
                    }).show();
        }
    }
    /**
     * 保存本地照片
     *
     * @param bmp
     * @return
     */
    public static File saveImage(Bitmap bmp) {
        File appDir = new File(Environment.getExternalStorageDirectory(), "images");
        if (!appDir.exists()) {
            appDir.mkdir();
        }
        String fileName = System.currentTimeMillis() + ".png";
        File file = new File(appDir, fileName);
        try {
            FileOutputStream fos = new FileOutputStream(file);
            bmp.compress(Bitmap.CompressFormat.JPEG, 100, fos);
            fos.flush();
            fos.close();
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return file;
    }
    /**
     * 點擊事件
     **/
    public void startCamera() {
        parameters = mCamera.getParameters();
        parameters.setPictureFormat(ImageFormat.JPEG);
        parameters.setPictureSize(800, 400);
        parameters.setFocusMode(Camera.Parameters.FOCUS_MODE_AUTO);
        mCamera.autoFocus(new Camera.AutoFocusCallback() {
            @Override
            public void onAutoFocus(boolean success, Camera camera) {
                if (success) {
                    mCamera.takePicture(shutterCallback, null, mPictureCallback);
                }
            }
        });
    }
   //相機的啓動和Activity的生命週期進行綁定
    @Override
    protected void onResume() {
        super.onResume();
        if (mCamera == null) {
            mCamera = getCamera();
            if (mHolder != null) {
                setStartPrevicw(mCamera, mHolder);
            }
        }
    }
    //相機的啓動和Activity的生命週期進行綁定
    @Override
    protected void onPause() {
        super.onPause();
        releaseCamera();
    }
    /**
     * 打開攝像頭
     *
     * @return
     */
    public Camera getCamera() {
        Camera camera;
        try {
            camera = Camera.open();
        } catch (Exception e) {
            camera = null;
            e.printStackTrace();
        }
        return camera;
    }

    /**
     * 開始預覽相機內容
     */
    private void setStartPrevicw(Camera camera, SurfaceHolder holder) {
        try {
            camera.setPreviewDisplay(holder);
            //系統預覽角度的調整
            camera.setDisplayOrientation(90);
            //打開攝像頭
            camera.startPreview();
        } catch (IOException e) {
            e.printStackTrace();
            //釋放攝像頭
            releaseCamera();
        }
    }

    /**
     * 釋放相機資源
     */
    private void releaseCamera() {
        if (mCamera != null) {
            mCamera.setPreviewCallback(null);
            mCamera.stopPreview();
            mCamera.release();
            mCamera = null;
        }
    }

    //在開始的時候創建畫面顯示的東西
    @Override
    public void surfaceCreated(SurfaceHolder holder) {
        setStartPrevicw(mCamera, mHolder);
    }
    //當屏幕發生變化時候要做的事兒
    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
        mCamera.stopPreview();
        setStartPrevicw(mCamera, mHolder);
    }
    //當界面銷燬的時候要處理的事兒
    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {
        releaseCamera();
    }
}
下面我們要在manifest.xml中添加權限
<uses-permission android:name="android.permission.CAMERA" />
<uses-feature android:name="android.hardware.camera" />
<uses-feature android:name="android.hardware.camera.autofocus" />
就這樣,基本上就可以實現拍照效果了,體驗拍照帶給我們的樂趣吧;
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章