Search code examples
androidandroid-camerasurfaceview

SurfaceView how to Re-Create if it has been destroyed


I create a Camera by SurfaceView, But when I press home and go back to the activity again. The SurfaceView didn't show anything anymore. It didn't invoke surfaceCreated(SurfaceHolder holder)

while I back to the activity. What should I do?

here is my code write by Kotlin

class MyCameraView : SurfaceView , 
SurfaceHolder.Callback,Camera.PreviewCallback,Camera.FaceDetectionListener {

    private lateinit var mCamera : Camera
    var sizes : List<Camera.Size>? = null

    constructor( ctx : Context,  attrs : AttributeSet?) : super(ctx,attrs){
        holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
        holder.addCallback(this)
    }

    constructor(ctx : Context):this(ctx,null){}

    override fun surfaceChanged(p0: SurfaceHolder?, format: Int, width: Int, height: Int) {
        Log.e("MyCameraView","CHANGE ")
        val sizes = mCamera.parameters.supportedPreviewSizes

        val size = getOptimalSzie(sizes)

        val supportWith = size.height
        val supportHeight : Int = size.width
        val radio : Float  = width.toFloat()/supportWith
        setMeasuredDimension(width,(supportHeight*radio).toInt())
        layout(0, (-(supportHeight*radio)/5).toInt(),width, ((supportHeight*radio).toInt()-(supportHeight*radio)/5).toInt())


    }

    override fun surfaceDestroyed(p0: SurfaceHolder?) {
        Log.e("MyCameraView","DESTORY")
        holder.removeCallback(this)
        mCamera.setPreviewCallback(null)
        mCamera.stopFaceDetection()
        mCamera.stopPreview()
        mCamera.lock()
        mCamera.release()
    }

    override fun surfaceCreated(p0: SurfaceHolder?) {

        Log.e("MyCameraView","OPEN")

        mCamera = Camera.open(Camera.CameraInfo.CAMERA_FACING_FRONT)
        mCamera.setDisplayOrientation(90)
        mCamera.setPreviewCallback(this)
        mCamera.setPreviewDisplay(holder)

        mCamera.setFaceDetectionListener(this)
        mCamera.startPreview();
        mCamera.startFaceDetection()
    }

    private fun getOptimalSzie(sizes: List<Camera.Size>): Camera.Size {
        var pos = 0
        var ratio = 0
        var viewRatio = height/width
        sizes.forEachIndexed { index, size ->
            val curRatio = size.width/size.height
            if(ratio == 0 ) ratio = curRatio
            else if( (viewRatio - curRatio) < (viewRatio - ratio)  ){
                ratio = curRatio
                pos = index
            }
        }
        return sizes[pos]
    }

    override fun onPreviewFrame(data: ByteArray?, camera: Camera) {
        if(allowTake) {

            val parameters = camera.getParameters()
            val width = parameters.previewSize.width
            val height = parameters.previewSize.height

            val yuv = YuvImage(data, parameters.previewFormat, width, height, null)

            val out = ByteArrayOutputStream()
            yuv.compressToJpeg(Rect(0, 0, width, height), 50, out)
            val bytes = out.toByteArray()
            val b = BitmapFactory.decodeByteArray(bytes, 0, bytes.size)
            detected?.invoke(bytes)
            allowTake = false
            hasTake = true
        }
    }

    var hasTake = false
    private var allowTake = false
    override fun onFaceDetection(faces: Array<out Camera.Face>?, camera: Camera?) {
        if(!hasTake) {
            allowTake = true
        }
    }

    private var detected: ((bitmap : ByteArray) -> Unit)? = null

    fun onFaceDetected(detected : ( bitmap : ByteArray )->Unit){
        this.detected = detected
    }

}

Solution

  • I slove it! I change to use TextureView. And it has four callback method. here is one of them

    onSurfaceTextureAvailable(surface: SurfaceTexture?, width: Int, height: Int)

    it was called while i back to the activity!!! so i can open the camera when i back again