// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.

import UIKit
import PromiseKit

class MediaDismissAnimationController: NSObject {
    private let mediaItem: Media
    public let interactionController: MediaInteractiveDismiss?

    var fromView: UIView?
    var transitionView: UIView?
    var fromTransitionalOverlayView: UIView?
    var toTransitionalOverlayView: UIView?
    var fromMediaFrame: CGRect?
    var pendingCompletion: (() -> ())?

    init(galleryItem: MediaGalleryViewModel.Item, interactionController: MediaInteractiveDismiss? = nil) {
        self.mediaItem = .gallery(galleryItem)
        self.interactionController = interactionController
    }

    init(image: UIImage, interactionController: MediaInteractiveDismiss? = nil) {
        self.mediaItem = .image(image)
        self.interactionController = interactionController
    }
}

extension MediaDismissAnimationController: UIViewControllerAnimatedTransitioning {
    func transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval {
        return 0.3
    }

    func animateTransition(using transitionContext: UIViewControllerContextTransitioning) {
        let containerView = transitionContext.containerView
        let fromContextProvider: MediaPresentationContextProvider
        let toContextProvider: MediaPresentationContextProvider

        guard let fromVC: UIViewController = transitionContext.viewController(forKey: .from) else {
            transitionContext.completeTransition(false)
            return
        }
        guard let toVC: UIViewController = transitionContext.viewController(forKey: .to) else {
            transitionContext.completeTransition(false)
            return
        }

        switch fromVC {
            case let contextProvider as MediaPresentationContextProvider:
                fromContextProvider = contextProvider

            case let navController as UINavigationController:
                guard let contextProvider = navController.topViewController as? MediaPresentationContextProvider else {
                    transitionContext.completeTransition(false)
                    return
                }

                fromContextProvider = contextProvider

            default:
                transitionContext.completeTransition(false)
                return
        }

        switch toVC {
            case let contextProvider as MediaPresentationContextProvider:
                toVC.view.layoutIfNeeded()
                toContextProvider = contextProvider

            case let navController as UINavigationController:
                guard let contextProvider = navController.topViewController as? MediaPresentationContextProvider else {
                    transitionContext.completeTransition(false)
                    return
                }

                toVC.view.layoutIfNeeded()
                toContextProvider = contextProvider

            default:
                transitionContext.completeTransition(false)
                return
        }

        guard let fromMediaContext: MediaPresentationContext = fromContextProvider.mediaPresentationContext(mediaItem: mediaItem, in: containerView) else {
            transitionContext.completeTransition(false)
            return
        }

        guard let presentationImage: UIImage = mediaItem.image else {
            transitionContext.completeTransition(true)
            return
        }

        // fromView will be nil if doing a presentation, in which case we don't want to add the view -
        // it will automatically be added to the view hierarchy, in front of the VC we're presenting from
        if let fromView: UIView = transitionContext.view(forKey: .from) {
            self.fromView = fromView
            containerView.addSubview(fromView)
        }

        // toView will be nil if doing a modal dismiss, in which case we don't want to add the view -
        // it's already in the view hierarchy, behind the VC we're dismissing.
        if let toView: UIView = transitionContext.view(forKey: .to) {
            containerView.insertSubview(toView, at: 0)
        }

        let toMediaContext: MediaPresentationContext? = toContextProvider.mediaPresentationContext(mediaItem: mediaItem, in: containerView)
        let duration: CGFloat = transitionDuration(using: transitionContext)

        fromMediaContext.mediaView.alpha = 0
        toMediaContext?.mediaView.alpha = 0

        let transitionView = UIImageView(image: presentationImage)
        transitionView.frame = fromMediaContext.presentationFrame
        transitionView.contentMode = MediaView.contentMode
        transitionView.layer.masksToBounds = true
        transitionView.layer.cornerRadius = fromMediaContext.cornerRadius
        transitionView.layer.maskedCorners = (toMediaContext?.cornerMask ?? fromMediaContext.cornerMask)
        containerView.addSubview(transitionView)

        // Add any UI elements which should appear above the media view
        self.fromTransitionalOverlayView = {
            guard let (overlayView, overlayViewFrame) = fromContextProvider.snapshotOverlayView(in: containerView) else {
                return nil
            }

            overlayView.frame = overlayViewFrame
            containerView.addSubview(overlayView)

            return overlayView
        }()
        self.toTransitionalOverlayView = { [weak self] in
            guard let (overlayView, overlayViewFrame) = toContextProvider.snapshotOverlayView(in: containerView) else {
                return nil
            }

            // Only fade in the 'toTransitionalOverlayView' if it's bigger than the origin
            // one (makes it look cleaner as you don't get the crossfade effect)
            if (self?.fromTransitionalOverlayView?.frame.size.height ?? 0) > overlayViewFrame.height {
                overlayView.alpha = 0
            }

            overlayView.frame = overlayViewFrame

            if let fromTransitionalOverlayView = self?.fromTransitionalOverlayView {
                containerView.insertSubview(overlayView, belowSubview: fromTransitionalOverlayView)
            }
            else {
                containerView.addSubview(overlayView)
            }

            return overlayView
        }()

        self.transitionView = transitionView
        self.fromMediaFrame = transitionView.frame

        self.pendingCompletion = {
            let destinationFromAlpha: CGFloat
            let destinationFrame: CGRect
            let destinationCornerRadius: CGFloat

            if transitionContext.transitionWasCancelled {
                destinationFromAlpha = 1
                destinationFrame = fromMediaContext.presentationFrame
                destinationCornerRadius = fromMediaContext.cornerRadius
            }
            else if let toMediaContext: MediaPresentationContext = toMediaContext {
                destinationFromAlpha = 0
                destinationFrame = toMediaContext.presentationFrame
                destinationCornerRadius = toMediaContext.cornerRadius
            }
            else {
                // `toMediaContext` can be nil if the target item is scrolled off of the
                // contextProvider's screen, so we synthesize a context to dismiss the item
                // off screen
                destinationFromAlpha = 0
                destinationFrame = fromMediaContext.presentationFrame
                    .offsetBy(dx: 0, dy: (containerView.bounds.height * 2))
                destinationCornerRadius = fromMediaContext.cornerRadius
            }

            UIView.animate(
                withDuration: duration,
                delay: 0,
                options: [.beginFromCurrentState, .curveEaseInOut],
                animations: { [weak self] in
                    self?.fromTransitionalOverlayView?.alpha = destinationFromAlpha
                    self?.fromView?.alpha = destinationFromAlpha
                    self?.toTransitionalOverlayView?.alpha = (1.0 - destinationFromAlpha)
                    transitionView.frame = destinationFrame
                    transitionView.layer.cornerRadius = destinationCornerRadius
                },
                completion: { [weak self] _ in
                    self?.fromView?.alpha = 1
                    fromMediaContext.mediaView.alpha = 1
                    toMediaContext?.mediaView.alpha = 1
                    transitionView.removeFromSuperview()
                    self?.fromTransitionalOverlayView?.removeFromSuperview()
                    self?.toTransitionalOverlayView?.removeFromSuperview()

                    if transitionContext.transitionWasCancelled {
                        // The "to" view will be nil if we're doing a modal dismiss, in which case
                        // we wouldn't want to remove the toView.
                        transitionContext.view(forKey: .to)?.removeFromSuperview()
                        
                        // Note: We shouldn't need to do this but for some reason it's not
                        // automatically getting re-enabled so we manually enable it
                        transitionContext.view(forKey: .from)?.isUserInteractionEnabled = true
                    }
                    else {
                        transitionContext.view(forKey: .from)?.removeFromSuperview()
                        
                        // Note: We shouldn't need to do this but for some reason it's not
                        // automatically getting re-enabled so we manually enable it
                        transitionContext.view(forKey: .to)?.isUserInteractionEnabled = true
                    }

                    transitionContext.completeTransition(!transitionContext.transitionWasCancelled)
                }
            )
        }

        // The interactive transition will call the 'pendingCompletion' when it completes so don't call it here
        guard !transitionContext.isInteractive else { return }

        self.pendingCompletion?()
        self.pendingCompletion = nil
    }
}

extension MediaDismissAnimationController: InteractiveDismissDelegate {
    func interactiveDismissUpdate(_ interactiveDismiss: UIPercentDrivenInteractiveTransition, didChangeTouchOffset offset: CGPoint) {
        guard let transitionView: UIView = transitionView else { return } // Transition hasn't started yet
        guard let fromMediaFrame: CGRect = fromMediaFrame else { return }

        fromView?.alpha = (1.0 - interactiveDismiss.percentComplete)
        transitionView.center = fromMediaFrame.offsetBy(dx: offset.x, dy: offset.y).center
    }

    func interactiveDismissDidFinish(_ interactiveDismiss: UIPercentDrivenInteractiveTransition) {
        self.pendingCompletion?()
        self.pendingCompletion = nil
    }
}