Приложение вылетает при повороте экрана
Я использую Mobile Vision API для обнаружения лиц на передней камере. Я использовал демонстрацию Google FaceTracker, но мне нужно использовать ее во Fragment. Это работает, но когда я пытаюсь повернуть экран, приложение вылетает за исключением RuntimeException: Fail to connect to camera service
,
Код фрагмента:
class EyeTrackerFragment : Fragment(), AnkoLogger {
private var cameraSource: CameraSource? = null
private lateinit var cameraSourcePreview: CameraSourcePreview
private lateinit var graphicOverlay: GraphicOverlay
private var callback: OnEyeContextUpdatedListener? = null
companion object {
// google play services error code
private val RC_HANDLE_GMS = 9001
// permission request codes need to be < 256
private val RC_HANDLE_CAMERA_PERM = 2
}
override fun onAttach(context: Context?) {
super.onAttach(context)
try {
callback = activity as OnEyeContextUpdatedListener
} catch (e: ClassCastException) {
throw ClassCastException("${activity} must implement OnEyeContextUpdatedListener")
}
}
/**
* Restarts the camera.
*/
override fun onResume() {
super.onResume()
startCameraSource()
}
/**
* Stops the camera.
*/
override fun onPause() {
super.onPause()
cameraSourcePreview.stop()
}
/**
* Releases the resources associated with the camera source, the associated detector, and the
* rest of the processing pipeline.
*/
override fun onDestroy() {
super.onDestroy()
cameraSource?.release()
}
override fun onCreateView(inflater: LayoutInflater?, container: ViewGroup?, savedInstanceState: Bundle?): View? =
inflater!!.inflate(R.layout.fragment_face_tracker, container, false)
override fun onViewCreated(view: View?, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
cameraSourcePreview = getView()!!.findViewById<View>(R.id.preview) as CameraSourcePreview
graphicOverlay = getView()!!.findViewById<View>(R.id.faceOverlay) as GraphicOverlay
if (ActivityCompat.checkSelfPermission(context.applicationContext, Manifest.permission.CAMERA) == PackageManager.PERMISSION_GRANTED) {
createCameraSource()
} else {
requestCameraPermission()
}
}
/**
* Handles the requesting of the camera permission. This includes
* showing a "Toast" message of why the permission is needed then
* sending the request.
*/
private fun requestCameraPermission() {
warn("Camera permission is not granted. Requesting permission")
val permissions = arrayOf(Manifest.permission.CAMERA)
if (!ActivityCompat.shouldShowRequestPermissionRationale(activity,
Manifest.permission.CAMERA)) {
ActivityCompat.requestPermissions(activity, permissions, RC_HANDLE_CAMERA_PERM)
return
}
Toast.makeText(context.applicationContext, "Give me permissions", Toast.LENGTH_LONG).show()
}
private fun createCameraSource() {
val detector = FaceDetector.Builder(context.applicationContext)
.setClassificationType(FaceDetector.ALL_CLASSIFICATIONS)
.setProminentFaceOnly(true)
.setTrackingEnabled(true)
.setMode(FaceDetector.ACCURATE_MODE)
.build()
val faceTracker = GraphicFaceTracker(graphicOverlay, callback, context)
val faceProcessor = LargestFaceFocusingProcessor(detector, faceTracker)
detector.setProcessor(faceProcessor)
if (!detector.isOperational) {
longToast("Face detector dependencies are not yet available.")
}
cameraSource = CameraSource.Builder(context, detector)
.setRequestedPreviewSize(640, 480)
.setFacing(CameraSource.CAMERA_FACING_FRONT)
.setRequestedFps(30.0f)
.build()
}
private fun startCameraSource() {
// check that the device has play services available.
val code = GoogleApiAvailability.getInstance().isGooglePlayServicesAvailable(
context.applicationContext)
if (code != ConnectionResult.SUCCESS) {
val dlg = GoogleApiAvailability.getInstance().getErrorDialog(activity, code, RC_HANDLE_GMS)
dlg.show()
}
if (cameraSource != null) {
try {
cameraSourcePreview.start(cameraSource!!, graphicOverlay)
} catch (e: IOException) {
error("Unable to start camera source.", e)
cameraSource!!.release()
cameraSource = null
}
}
}
override fun onRequestPermissionsResult(requestCode: Int, permissions: Array<String>, grantResults: IntArray) {
if (requestCode != RC_HANDLE_CAMERA_PERM) {
debug("Got unexpected permission result: $requestCode")
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
return
}
if (grantResults.size != 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
debug("Camera permission granted - initialize the camera source")
createCameraSource()
return
}
error("Permission not granted: results len = ${grantResults.size}. Result code = ${if (grantResults.isNotEmpty()) grantResults[0] else "(empty)"}")
val builder = AlertDialog.Builder(context)
builder.setTitle("Face Tracker sample")
.setMessage("Have no camera permission")
.setPositiveButton("Ok", { _, _ -> activity.finish() })
.show()
}
}
Согласно коду активности. Он только присоединяет предыдущий фрагмент в методе onCreate, а затем ничего не делает.
Я думаю, что экземпляр камеры не был выпущен в некоторых ситуациях, но это делается так же, как в Activity. Есть ли какое-либо специальное поведение во Фрагменте, которое потребовало бы дополнительных действий по сравнению с активами?
Полный журнал исключений:
E/AndroidRuntime: FATAL EXCEPTION: main
Process: com.eyetracker.android.demo, PID: 22631
java.lang.RuntimeException: Fail to connect to camera service
at android.hardware.Camera.<init>(Camera.java:520)
at android.hardware.Camera.open(Camera.java:361)
at com.google.android.gms.vision.CameraSource.zzchq(Unknown Source)
at com.google.android.gms.vision.CameraSource.start(Unknown Source)
at com.eyetracker.android.camera.CameraSourcePreview.startIfReady(CameraSourcePreview.kt:82)
at com.eyetracker.android.camera.CameraSourcePreview.access$startIfReady(CameraSourcePreview.kt:18)
at com.eyetracker.android.camera.CameraSourcePreview$SurfaceCallback.surfaceCreated(CameraSourcePreview.kt:104)
at android.view.SurfaceView.updateWindow(SurfaceView.java:580)
at android.view.SurfaceView$3.onPreDraw(SurfaceView.java:176)
at android.view.ViewTreeObserver.dispatchOnPreDraw(ViewTreeObserver.java:948)
at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:1974)
at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1065)
at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:5901)
at android.view.Choreographer$CallbackRecord.run(Choreographer.java:767)
at android.view.Choreographer.doCallbacks(Choreographer.java:580)
at android.view.Choreographer.doFrame(Choreographer.java:550)
at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:753)
at android.os.Handler.handleCallback(Handler.java:739)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:211)
at android.app.ActivityThread.main(ActivityThread.java:5389)
at java.lang.reflect.Method.invoke(Native Method)
at java.lang.reflect.Method.invoke(Method.java:372)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1020)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:815)
1 ответ
Всякий раз, когда Android разрушает и воссоздает вашу активность для изменения ориентации, он вызывает метод SaveInstanceState().
Так что сохраните необходимые данные в Bundle в onSaveInstanceState() и восстановите то же самое в onCreate(), проверив, является ли поле SavedInstanceState пустым?