2016-10-06 147 views
0

我不能理解爲什麼這不起作用。iOS交互式動畫Swift

有兩名風險投資:A和B.

我想向右滑動VC的一個揭示VC B,而要使它互動,使用戶可以在兩個風投之間拖動(像Instagram的主屏幕上時,你左右滑動去相機和信息)。目前,它不會「拖拽」。您可以刷卡上VC A和它會去VC B.

這裏是我的動畫對象向右滑動:

class SlideRightTransitionAnimator: NSObject { 

let duration = 0.5 
var isPresenting = false 
let customInteractiveTransition = CustomInteractiveTransition() 

} 

// MARK: UIViewControllerTransitioningDelegate 
extension SlideRightTransitionAnimator: UIViewControllerTransitioningDelegate { 

// Return the animator when presenting a VC 
func animationController(forPresented presented: UIViewController, presenting: UIViewController, source: UIViewController) -> UIViewControllerAnimatedTransitioning? { 
    isPresenting = true 
    return self 
} 

// Return the animator used when dismissing from a VC 
func animationController(forDismissed dismissed: UIViewController) -> UIViewControllerAnimatedTransitioning? { 
    isPresenting = false 
    return self 
} 

// Add the interactive transition for Presentation only 
func interactionControllerForPresentation(using animator: UIViewControllerAnimatedTransitioning) -> UIViewControllerInteractiveTransitioning? { 
    return customInteractiveTransition.transitionInProgress ? customInteractiveTransition : nil 
} 
} 

// MARK: UIViewControllerTransitioningDelegate 
extension SlideRightTransitionAnimator: UIViewControllerAnimatedTransitioning { 

// Return how many seconds the transiton animation will take 
func transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval { 
    return duration 
} 

// Animate a change from one VC to another 
func animateTransition(using transitionContext: UIViewControllerContextTransitioning) { 

    // Get reference to our fromView, toView and the container view that we should perform the transition 
    let container = transitionContext.containerView 
    let fromView = transitionContext.view(forKey: UITransitionContextViewKey.from)! 
    let toView = transitionContext.view(forKey: UITransitionContextViewKey.to)! 

    // Set up the transform we'll use in the animation 
    let offScreenRight = CGAffineTransform(translationX: container.frame.width, y: 0) 
    let offScreenLeft = CGAffineTransform(translationX: -container.frame.width, y: 0) 

    // Start the toView to the right of the screen 
    if isPresenting { 
     toView.transform = offScreenLeft 
    } 

    // Add the both views to our VC 
    container.addSubview(toView) 
    container.addSubview(fromView) 

    // Perform the animation 
    UIView.animate(withDuration: duration, delay: 0.0, usingSpringWithDamping: 1.0, initialSpringVelocity: 1.0, options: [], animations: { 

     if self.isPresenting { 
      fromView.transform = offScreenRight 
      toView.transform = CGAffineTransform.identity 
     } 
     else { 
      fromView.transform = offScreenLeft 
      toView.transform = CGAffineTransform.identity 
     } 

     }, completion: { finished in 
      // Tell our transitionContext object that we've finished animating 
      transitionContext.completeTransition(true) 
    }) 
} 

} 

這裏是我的互動轉變控制器

class CustomInteractiveTransition: UIPercentDrivenInteractiveTransition { 

weak var viewController : UIViewController! 
var shouldCompleteTransition = false 
var transitionInProgress = false 
var completionSeed: CGFloat { 
    return 1 - percentComplete 
} 

func attachToViewController(viewController: UIViewController) { 
    self.viewController = viewController 
    setupPanGestureRecognizer(view: viewController.view) 
} 

private func setupPanGestureRecognizer(view: UIView) { 
    view.addGestureRecognizer(UIPanGestureRecognizer(target: self, action: #selector(handlePanGesture))) 
} 

func handlePanGesture(gestureRecognizer: UIPanGestureRecognizer) { 
    let viewTranslation = gestureRecognizer.translation(in: gestureRecognizer.view!.superview!) 
    var progress = (viewTranslation.x/200) 
    progress = CGFloat(fminf(fmaxf(Float(progress), 0.0), 1.0)) 

    switch gestureRecognizer.state { 
    case .began: 
     transitionInProgress = true 
     viewController.dismiss(animated: true, completion: nil) 
    case .changed: 
     shouldCompleteTransition = progress > 0.5 
     update(progress) 
    case .cancelled, .ended: 
     transitionInProgress = false 
     if !shouldCompleteTransition || gestureRecognizer.state == .cancelled { 
      cancel() 
     } else { 
      finish() 
     } 
    default: 
     print("Swift switch must be exhaustive, thus the default") 
    } 
} 

} 

而且最後的代碼VC - 答:

class ViewControllerA: UITableViewController { 

let slideRightTransition = SlideRightTransitionAnimator() 
let customInteractiveTransition = CustomInteractiveTransition() 

override func viewDidLoad() { 
    super.viewDidLoad() 

    // Add a Pan Gesture to swipe to other VC 
    let swipeGestureRight = UISwipeGestureRecognizer(target: self, action: #selector(swipeGestureRightAction)) 
    swipeGestureRight.direction = .right 
    view.addGestureRecognizer(swipeGestureRight) 
} 

    // MARK: Pan gestures 
func swipeGestureRightAction() { 
    performSegue(withIdentifier: "showMapSegue", sender: self) 
} 

// MARK: Prepare for segue 
override func prepare(for segue: UIStoryboardSegue, sender: Any?) { 

    if segue.identifier == "showVCB" { 

     // This gets a reference to the screen that we're about to transition to and from 
     let toViewController = segue.destination as UIViewController 

     // Instead of using the default transition animation, we'll ask the segue to use our custom SlideRightTransitionAnimator object to manage the transition animation 
     toViewController.transitioningDelegate = slideRightTransition 

     // Add the Interactive gesture transition to the VC 
     customInteractiveTransition.attachToViewController(viewController: toViewController) 
    } 
} 

謝謝提前!

回答

2

在VC A的viewDidLoad中,您需要用UIPanGestureRecognizer替換該UISwipeGestureRecognizer。然後在手勢處理函數中實現適當的.changed狀態。編輯:此外,要實現控制器之間的滑動轉換,我強烈建議使用UIPageViewController來代替。這或者甚至可以是自定義的UICollectionView解決方案。

+0

很抱歉這麼晚纔回復,但肯定你是對的 - 關鍵是要使用UIPageViewController。謝謝 – Danny

0
在功能

handlePanGesture你是從一個VC服用翻譯參考(即,gestureRecognizer.view!.superview!) 但而不是採取全球化志願服務青年從一個UIWindow。 (即,UIApplication.shared.windows.last

更換gestureRecognizer.view!.superview!UIApplication.shared.windows.last

it's worked for me.