后退按钮启动UIViewPropertyAnimator?

时间:2018-06-13 20:00:30

标签: ios swift uiviewanimation uiviewpropertyanimator

我有一个带摄像头的ViewController,用于录制视频。在顶部有一个旋转圆圈,表示正在录制视频。这样设置如下:

class CameraViewController: UIViewController, UIImagePickerControllerDelegate, UINavigationControllerDelegate {

    private var animator: UIViewPropertyAnimator?

    @objc func handleTap(_ gesture:UITapGestureRecognizer) {
        if animator == nil {
            createAnimation()
        }
        startRecording()
    }
    private func createAnimation() {
        animator = UIViewPropertyAnimator.runningPropertyAnimator(withDuration: 4, delay: 0, options: [.curveLinear,.allowUserInteraction], animations: {
            UIView.animateKeyframes(withDuration: 4, delay: 0, animations: {
                UIView.addKeyframe(withRelativeStartTime: 0, relativeDuration: 1.0 / 3.0) {
                    self.recordingSpinner.transform = .init(rotationAngle: .pi * 2 * 1 / 3)
                }
                UIView.addKeyframe(withRelativeStartTime: 1.0 / 3.0, relativeDuration: 1.0 / 3.0) {
                    self.recordingSpinner.transform = .init(rotationAngle: .pi * 2 * 2 / 3)
                }
                UIView.addKeyframe(withRelativeStartTime: 2.0 / 3.0, relativeDuration: 1.0 / 3.0) {
                    self.recordingSpinner.transform = .identity
                }
            })
        }, completion: { [weak self] _ in
            self?.createAnimation()
        })
    }

    func startRecording() {

        if movieOutput.isRecording == false {

            animator?.startAnimation()
            let connection = movieOutput.connection(with: AVMediaType.video)
            if (connection?.isVideoOrientationSupported)! {
                connection?.videoOrientation = currentVideoOrientation()
            }

            if (connection?.isVideoStabilizationSupported)! {
                connection?.preferredVideoStabilizationMode = AVCaptureVideoStabilizationMode.auto
            }

            let device = activeInput.device
            if (device.isSmoothAutoFocusSupported) {
                do {
                    try device.lockForConfiguration()
                    device.isSmoothAutoFocusEnabled = false
                    device.unlockForConfiguration()
                } catch {
                    print("Error setting configuration: \(error)")
                }

            }

            let outputFileName = NSUUID().uuidString
            let outputFilePath = (NSTemporaryDirectory() as NSString).appendingPathComponent((outputFileName as NSString).appendingPathExtension("mov")!)
            movieOutput.startRecording(to: URL(fileURLWithPath: outputFilePath), recordingDelegate: self)

        }
        else {
            stopRecording()
        }
    }

    func stopRecording() {
        if movieOutput.isRecording == true {
            animator?.pauseAnimation()
            movieOutput.stopRecording()
        }
    }
    @IBAction func unwindToCamera(sender: UIStoryboardSegue) {
    }
    ...
}

extension CameraViewController: AVCaptureFileOutputRecordingDelegate{
    func fileOutput(_ output: AVCaptureFileOutput, didFinishRecordingTo outputFileURL: URL, from connections: [AVCaptureConnection], error: Error?) {
        if (error != nil) {
            print("Error recording movie: \(error!.localizedDescription)")
        } else {
            self.footageURL = outputFileURL as URL
            //print(self.videoRecorded!)
            self.performSegue(withIdentifier: "TrimFootage_Segue", sender: nil)
        }
    }
    override func prepare(for segue: UIStoryboardSegue, sender: Any?){
        if segue.identifier == "TrimFootage_Segue" {
            let controller = segue.destination as! TrimFootageViewController
            controller.footageURL = self.footageURL
        }
    }
}

所以它创建一个动画师,如果它不存在,然后调用startRecording开始动画。然后stopRecording停止它。然后,当视频完成录制到输出文件时,它将转移到newView控制器。当我按下该视图控制器时,它使用了展开segue - unwindToCameraWithSender:

当我放松并回到相机时,视频没有录制,但动画正在播放。什么可能导致这个动画再次开始?我怎么能阻止这个?

2 个答案:

答案 0 :(得分:1)

我认为动画只是暂停是原因。在Callvirt方法中尝试

stopRecording()

而不是

animator?.stopAnimation(true)

答案 1 :(得分:0)

所以我做了一件事来解决这个问题,但没有解决这个问题就像使用UIDynamicAnimator一样:

@objc func handleTap(_ gesture:UITapGestureRecognizer) {
        startRecording()

        if let rotate = rotate{
            animator.removeBehavior(rotate)
            self.rotate = nil
        } else {
            rotate = UIDynamicItemBehavior(items: [self.recordingSpinner])
            rotate?.allowsRotation = true
            rotate?.angularResistance = 0
            rotate?.addAngularVelocity(1, for: self.recordingSpinner)
            animator.addBehavior(rotate!)
        }
    }

取自这个答案:Proper way to stop an infinitely rotating image? and how does one implement removeAllAnimations?

有趣的是,当我执行segue时似乎没有开始旋转,虽然我不知道为什么。如果有人想到我为什么喜欢听他们的话