[
  {
    "path": ".gitignore",
    "content": "# Xcode\n#\n# gitignore contributors: remember to update Global/Xcode.gitignore, Objective-C.gitignore & Swift.gitignore\n\n## User settings\nxcuserdata/\n\n## compatibility with Xcode 8 and earlier (ignoring not required starting Xcode 9)\n*.xcscmblueprint\n*.xccheckout\n\n## compatibility with Xcode 3 and earlier (ignoring not required starting Xcode 4)\nbuild/\nDerivedData/\n*.moved-aside\n*.pbxuser\n!default.pbxuser\n*.mode1v3\n!default.mode1v3\n*.mode2v3\n!default.mode2v3\n*.perspectivev3\n!default.perspectivev3\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/AppDelegate.swift",
    "content": "//\n//  AppDelegate.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\n@UIApplicationMain\nclass AppDelegate: UIResponder, UIApplicationDelegate {\n\n    var window: UIWindow?\n\n\n    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {\n        // Override point for customization after application launch.\n        return true\n    }\n\n    func applicationWillResignActive(_ application: UIApplication) {\n        // Sent when the application is about to move from active to inactive state. This can occur for certain types of temporary interruptions (such as an incoming phone call or SMS message) or when the user quits the application and it begins the transition to the background state.\n        // Use this method to pause ongoing tasks, disable timers, and invalidate graphics rendering callbacks. Games should use this method to pause the game.\n    }\n\n    func applicationDidEnterBackground(_ application: UIApplication) {\n        // Use this method to release shared resources, save user data, invalidate timers, and store enough application state information to restore your application to its current state in case it is terminated later.\n        // If your application supports background execution, this method is called instead of applicationWillTerminate: when the user quits.\n    }\n\n    func applicationWillEnterForeground(_ application: UIApplication) {\n        // Called as part of the transition from the background to the active state; here you can undo many of the changes made on entering the background.\n    }\n\n    func applicationDidBecomeActive(_ application: UIApplication) {\n        // Restart any tasks that were paused (or not yet started) while the application was inactive. If the application was previously in the background, optionally refresh the user interface.\n    }\n\n    func applicationWillTerminate(_ application: UIApplication) {\n        // Called when the application is about to terminate. Save data if appropriate. See also applicationDidEnterBackground:.\n    }\n\n\n}\n\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/Assets.xcassets/AppIcon.appiconset/Contents.json",
    "content": "{\n  \"images\" : [\n    {\n      \"idiom\" : \"iphone\",\n      \"size\" : \"20x20\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"iphone\",\n      \"size\" : \"20x20\",\n      \"scale\" : \"3x\"\n    },\n    {\n      \"idiom\" : \"iphone\",\n      \"size\" : \"29x29\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"iphone\",\n      \"size\" : \"29x29\",\n      \"scale\" : \"3x\"\n    },\n    {\n      \"idiom\" : \"iphone\",\n      \"size\" : \"40x40\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"iphone\",\n      \"size\" : \"40x40\",\n      \"scale\" : \"3x\"\n    },\n    {\n      \"idiom\" : \"iphone\",\n      \"size\" : \"60x60\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"iphone\",\n      \"size\" : \"60x60\",\n      \"scale\" : \"3x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"20x20\",\n      \"scale\" : \"1x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"20x20\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"29x29\",\n      \"scale\" : \"1x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"29x29\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"40x40\",\n      \"scale\" : \"1x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"40x40\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"76x76\",\n      \"scale\" : \"1x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"76x76\",\n      \"scale\" : \"2x\"\n    },\n    {\n      \"idiom\" : \"ipad\",\n      \"size\" : \"83.5x83.5\",\n      \"scale\" : \"2x\"\n    }\n  ],\n  \"info\" : {\n    \"version\" : 1,\n    \"author\" : \"xcode\"\n  }\n}"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/Base.lproj/LaunchScreen.storyboard",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"no\"?>\n<document type=\"com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB\" version=\"3.0\" toolsVersion=\"13122.16\" systemVersion=\"17A277\" targetRuntime=\"iOS.CocoaTouch\" propertyAccessControl=\"none\" useAutolayout=\"YES\" launchScreen=\"YES\" useTraitCollections=\"YES\" useSafeAreas=\"YES\" colorMatched=\"YES\" initialViewController=\"01J-lp-oVM\">\n    <dependencies>\n        <plugIn identifier=\"com.apple.InterfaceBuilder.IBCocoaTouchPlugin\" version=\"13104.12\"/>\n        <capability name=\"Safe area layout guides\" minToolsVersion=\"9.0\"/>\n        <capability name=\"documents saved in the Xcode 8 format\" minToolsVersion=\"8.0\"/>\n    </dependencies>\n    <scenes>\n        <!--View Controller-->\n        <scene sceneID=\"EHf-IW-A2E\">\n            <objects>\n                <viewController id=\"01J-lp-oVM\" sceneMemberID=\"viewController\">\n                    <view key=\"view\" contentMode=\"scaleToFill\" id=\"Ze5-6b-2t3\">\n                        <rect key=\"frame\" x=\"0.0\" y=\"0.0\" width=\"375\" height=\"667\"/>\n                        <autoresizingMask key=\"autoresizingMask\" widthSizable=\"YES\" heightSizable=\"YES\"/>\n                        <color key=\"backgroundColor\" red=\"1\" green=\"1\" blue=\"1\" alpha=\"1\" colorSpace=\"custom\" customColorSpace=\"sRGB\"/>\n                        <viewLayoutGuide key=\"safeArea\" id=\"6Tk-OE-BBY\"/>\n                    </view>\n                </viewController>\n                <placeholder placeholderIdentifier=\"IBFirstResponder\" id=\"iYj-Kq-Ea1\" userLabel=\"First Responder\" sceneMemberID=\"firstResponder\"/>\n            </objects>\n            <point key=\"canvasLocation\" x=\"53\" y=\"375\"/>\n        </scene>\n    </scenes>\n</document>\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/Base.lproj/Main.storyboard",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<document type=\"com.apple.InterfaceBuilder3.CocoaTouch.Storyboard.XIB\" version=\"3.0\" toolsVersion=\"13529\" targetRuntime=\"iOS.CocoaTouch\" propertyAccessControl=\"none\" useAutolayout=\"YES\" useTraitCollections=\"YES\" useSafeAreas=\"YES\" colorMatched=\"YES\" initialViewController=\"BYZ-38-t0r\">\n    <device id=\"retina4_7\" orientation=\"portrait\">\n        <adaptation id=\"fullscreen\"/>\n    </device>\n    <dependencies>\n        <deployment identifier=\"iOS\"/>\n        <plugIn identifier=\"com.apple.InterfaceBuilder.IBCocoaTouchPlugin\" version=\"13527\"/>\n        <capability name=\"Safe area layout guides\" minToolsVersion=\"9.0\"/>\n        <capability name=\"documents saved in the Xcode 8 format\" minToolsVersion=\"8.0\"/>\n    </dependencies>\n    <scenes>\n        <!--View Controller-->\n        <scene sceneID=\"tne-QT-ifu\">\n            <objects>\n                <viewController id=\"BYZ-38-t0r\" customClass=\"ViewController\" customModule=\"A_J_Full_Screen_Image_Browser\" customModuleProvider=\"target\" sceneMemberID=\"viewController\">\n                    <view key=\"view\" contentMode=\"scaleToFill\" id=\"8bC-Xf-vdC\">\n                        <rect key=\"frame\" x=\"0.0\" y=\"0.0\" width=\"375\" height=\"667\"/>\n                        <autoresizingMask key=\"autoresizingMask\" widthSizable=\"YES\" heightSizable=\"YES\"/>\n                        <subviews>\n                            <button opaque=\"NO\" contentMode=\"scaleToFill\" contentHorizontalAlignment=\"center\" contentVerticalAlignment=\"center\" buttonType=\"roundedRect\" lineBreakMode=\"middleTruncation\" translatesAutoresizingMaskIntoConstraints=\"NO\" id=\"4Se-Fg-AJ1\">\n                                <rect key=\"frame\" x=\"167\" y=\"318\" width=\"40\" height=\"30\"/>\n                                <state key=\"normal\" title=\"boom\"/>\n                                <connections>\n                                    <action selector=\"onButtonTapped:\" destination=\"BYZ-38-t0r\" eventType=\"touchUpInside\" id=\"cU7-4k-oyE\"/>\n                                </connections>\n                            </button>\n                        </subviews>\n                        <color key=\"backgroundColor\" red=\"1\" green=\"1\" blue=\"1\" alpha=\"1\" colorSpace=\"custom\" customColorSpace=\"sRGB\"/>\n                        <constraints>\n                            <constraint firstItem=\"4Se-Fg-AJ1\" firstAttribute=\"centerY\" secondItem=\"8bC-Xf-vdC\" secondAttribute=\"centerY\" id=\"4wX-6a-B2q\"/>\n                            <constraint firstItem=\"4Se-Fg-AJ1\" firstAttribute=\"centerX\" secondItem=\"8bC-Xf-vdC\" secondAttribute=\"centerX\" id=\"B2z-ho-Efc\"/>\n                        </constraints>\n                        <viewLayoutGuide key=\"safeArea\" id=\"6Tk-OE-BBY\"/>\n                    </view>\n                </viewController>\n                <placeholder placeholderIdentifier=\"IBFirstResponder\" id=\"dkx-z0-nzr\" sceneMemberID=\"firstResponder\"/>\n            </objects>\n        </scene>\n    </scenes>\n</document>\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/Info.plist",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<!DOCTYPE plist PUBLIC \"-//Apple//DTD PLIST 1.0//EN\" \"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">\n<plist version=\"1.0\">\n<dict>\n\t<key>CFBundleDevelopmentRegion</key>\n\t<string>$(DEVELOPMENT_LANGUAGE)</string>\n\t<key>CFBundleExecutable</key>\n\t<string>$(EXECUTABLE_NAME)</string>\n\t<key>CFBundleIdentifier</key>\n\t<string>$(PRODUCT_BUNDLE_IDENTIFIER)</string>\n\t<key>CFBundleInfoDictionaryVersion</key>\n\t<string>6.0</string>\n\t<key>CFBundleName</key>\n\t<string>$(PRODUCT_NAME)</string>\n\t<key>CFBundlePackageType</key>\n\t<string>APPL</string>\n\t<key>CFBundleShortVersionString</key>\n\t<string>1.0</string>\n\t<key>CFBundleVersion</key>\n\t<string>1</string>\n\t<key>LSRequiresIPhoneOS</key>\n\t<true/>\n\t<key>NSAppTransportSecurity</key>\n\t<dict>\n\t\t<key>NSAllowsArbitraryLoads</key>\n\t\t<true/>\n\t</dict>\n\t<key>UILaunchStoryboardName</key>\n\t<string>LaunchScreen</string>\n\t<key>UIMainStoryboardFile</key>\n\t<string>Main</string>\n\t<key>UIRequiredDeviceCapabilities</key>\n\t<array>\n\t\t<string>armv7</string>\n\t</array>\n\t<key>UISupportedInterfaceOrientations</key>\n\t<array>\n\t\t<string>UIInterfaceOrientationPortrait</string>\n\t\t<string>UIInterfaceOrientationLandscapeLeft</string>\n\t\t<string>UIInterfaceOrientationLandscapeRight</string>\n\t</array>\n\t<key>UISupportedInterfaceOrientations~ipad</key>\n\t<array>\n\t\t<string>UIInterfaceOrientationPortrait</string>\n\t\t<string>UIInterfaceOrientationPortraitUpsideDown</string>\n\t\t<string>UIInterfaceOrientationLandscapeLeft</string>\n\t\t<string>UIInterfaceOrientationLandscapeRight</string>\n\t</array>\n</dict>\n</plist>\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/ViewController.swift",
    "content": "//\n//  ViewController.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\nfinal class ViewController: UIViewController {\n\n    lazy var testVideo: MediaDownloadable = {\n        return SingleMedia(imageURL: URL(string: \"https://dummyimage.com/600&text=thumbnail\")!,\n                    isVideoThumbnail: true,\n                    videoURL: URL(string: \"http://jplayer.org/video/m4v/Big_Buck_Bunny_Trailer.m4v\")!)\n    }()\n\n    lazy var media: [MediaDownloadable] = {\n        return [testVideo,\n                SingleMedia(imageURL: URL(string: \"https://dummyimage.com/300\")!),\n                SingleMedia(imageURL: URL(string: \"https://dummyimage.com/600\")!),\n                testVideo]\n    }()\n\n    @IBAction func onButtonTapped(_ sender: UIButton) {\n        let vm = FullScreenImageBrowserViewModel(media: media)\n        let x = FullScreenImageBrowser(viewModel: vm)\n        present(x, animated: true, completion: nil)\n    }\n\n}\n\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/animator/FullScreenImageTransitionAnimator.swift",
    "content": "//\n//  TransitionAnimator.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\npublic final class FullScreenImageTransitionAnimator: NSObject, UIViewControllerAnimatedTransitioning {\n    var dismissing: Bool = false\n\n    var startingView: UIView?\n    var endingView: UIView?\n\n    var startingViewForAnimation: UIView?\n    var endingViewForAnimation: UIView?\n\n    var animationDurationWithZooming = 0.5\n    var animationDurationWithoutZooming = 0.3\n    var animationDurationFadeRatio = 4.0 / 9.0 {\n        didSet(value) {\n            animationDurationFadeRatio = min(value, 1.0)\n        }\n    }\n    var animationDurationEndingViewFadeInRatio = 0.1 {\n        didSet(value) {\n            animationDurationEndingViewFadeInRatio = min(value, 1.0)\n        }\n    }\n    var animationDurationStartingViewFadeOutRatio = 0.05 {\n        didSet(value) {\n            animationDurationStartingViewFadeOutRatio = min(value, 1.0)\n        }\n    }\n    var zoomingAnimationSpringDamping = 0.9\n\n    var shouldPerformZoomingAnimation: Bool {\n        return startingView != nil && endingView != nil\n    }\n\n    public func transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval {\n        if shouldPerformZoomingAnimation {\n            return animationDurationWithZooming\n        }\n        return animationDurationWithoutZooming\n    }\n\n    func fadeDurationForTransitionContext(_ transitionContext: UIViewControllerContextTransitioning) -> TimeInterval {\n        if shouldPerformZoomingAnimation {\n            return transitionDuration(using: transitionContext) * animationDurationFadeRatio\n        }\n        return transitionDuration(using: transitionContext)\n    }\n\n    // MARK: - UIViewControllerAnimatedTransitioning\n    public func animateTransition(using transitionContext: UIViewControllerContextTransitioning) {\n        setupTransitionContainerHierarchyWithTransitionContext(transitionContext)\n\n        if shouldPerformZoomingAnimation {\n            performZoomingAnimationWithTransitionContext(transitionContext)\n        }\n        performFadeAnimationWithTransitionContext(transitionContext)\n    }\n\n    func setupTransitionContainerHierarchyWithTransitionContext(_ transitionContext: UIViewControllerContextTransitioning) {\n\n        if let toView = transitionContext.view(forKey: UITransitionContextViewKey.to),\n            let toViewController = transitionContext.viewController(forKey: UITransitionContextViewControllerKey.to) {\n            toView.frame = transitionContext.finalFrame(for: toViewController)\n            let containerView = transitionContext.containerView\n\n            if !toView.isDescendant(of: containerView) {\n                containerView.addSubview(toView)\n            }\n        }\n\n        if dismissing {\n            if let fromView = transitionContext.view(forKey: UITransitionContextViewKey.from) {\n                transitionContext.containerView.bringSubview(toFront: fromView)\n            }\n        }\n    }\n\n    func performFadeAnimationWithTransitionContext(_ transitionContext: UIViewControllerContextTransitioning) {\n        let fadeView = dismissing ? transitionContext.view(forKey: UITransitionContextViewKey.from) : transitionContext.view(forKey: UITransitionContextViewKey.to)\n        let beginningAlpha: CGFloat = dismissing ? 1.0 : 0.0\n        let endingAlpha: CGFloat = dismissing ? 0.0 : 1.0\n\n        fadeView?.alpha = beginningAlpha\n\n        UIView.animate(withDuration: fadeDurationForTransitionContext(transitionContext), animations: { () -> Void in\n            fadeView?.alpha = endingAlpha\n        }) { _ in\n            if !self.shouldPerformZoomingAnimation {\n                self.completeTransitionWithTransitionContext(transitionContext)\n            }\n        }\n    }\n\n    func performZoomingAnimationWithTransitionContext(_ transitionContext: UIViewControllerContextTransitioning) {\n\n        let containerView = transitionContext.containerView\n        guard let startingView = startingView, let endingView = endingView else {\n            return\n        }\n        guard let startingViewForAnimation = self.startingViewForAnimation ?? self.startingView?.aj_snapshotView(),\n            let endingViewForAnimation = self.endingViewForAnimation ?? self.endingView?.aj_snapshotView() else {\n                return\n        }\n\n        let finalEndingViewTransform = endingView.transform\n        let endingViewInitialTransform = startingViewForAnimation.frame.height / endingViewForAnimation.frame.height\n        let translatedStartingViewCenter = startingView.aj_translatedCenterPointToContainerView(containerView)\n\n        startingViewForAnimation.center = translatedStartingViewCenter\n\n        endingViewForAnimation.transform = endingViewForAnimation.transform.scaledBy(x: endingViewInitialTransform, y: endingViewInitialTransform)\n        endingViewForAnimation.center = translatedStartingViewCenter\n        endingViewForAnimation.alpha = 0.0\n\n        containerView.addSubview(startingViewForAnimation)\n        containerView.addSubview(endingViewForAnimation)\n\n        endingView.alpha = 0.0\n        startingView.alpha = 0.0\n\n        let fadeInDuration = transitionDuration(using: transitionContext) * animationDurationEndingViewFadeInRatio\n        let fadeOutDuration = transitionDuration(using: transitionContext) * animationDurationStartingViewFadeOutRatio\n\n        UIView.animate(withDuration: fadeInDuration, delay: 0.0, options: [.allowAnimatedContent, .beginFromCurrentState], animations: { () -> Void in\n            endingViewForAnimation.alpha = 1.0\n        }) { _ in\n            UIView.animate(withDuration: fadeOutDuration, delay: 0.0, options: [.allowAnimatedContent, .beginFromCurrentState], animations: { () -> Void in\n                startingViewForAnimation.alpha = 0.0\n            }, completion: { _ in\n                startingViewForAnimation.removeFromSuperview()\n            })\n        }\n\n        let startingViewFinalTransform = 1.0 / endingViewInitialTransform\n        let translatedEndingViewFinalCenter = endingView.aj_translatedCenterPointToContainerView(containerView)\n\n        UIView.animate(withDuration: transitionDuration(using: transitionContext), delay: 0.0, usingSpringWithDamping:CGFloat(zoomingAnimationSpringDamping), initialSpringVelocity:0, options: [.allowAnimatedContent, .beginFromCurrentState], animations: { () -> Void in\n            endingViewForAnimation.transform = finalEndingViewTransform\n            endingViewForAnimation.center = translatedEndingViewFinalCenter\n            startingViewForAnimation.transform = startingViewForAnimation.transform.scaledBy(x: startingViewFinalTransform, y: startingViewFinalTransform)\n            startingViewForAnimation.center = translatedEndingViewFinalCenter\n\n        }) { _ in\n            endingViewForAnimation.removeFromSuperview()\n            endingView.alpha = 1.0\n            startingView.alpha = 1.0\n            self.completeTransitionWithTransitionContext(transitionContext)\n        }\n    }\n\n    func completeTransitionWithTransitionContext(_ transitionContext: UIViewControllerContextTransitioning) {\n        if transitionContext.isInteractive {\n            if transitionContext.transitionWasCancelled {\n                transitionContext.cancelInteractiveTransition()\n            } else {\n                transitionContext.finishInteractiveTransition()\n            }\n        }\n        transitionContext.completeTransition(!transitionContext.transitionWasCancelled)\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/core/FullScreenImageBrowser.swift",
    "content": "//\n//  FullScreenImageBrowser.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\nimport AVKit\n\npublic final class FullScreenImageBrowser: UIViewController {\n\n    // MARK: - Property\n    public var viewModel: FullScreenImageBrowserViewModel\n    public private(set) var pageViewController: UIPageViewController\n    public let transitionAnimator: FullScreenImageTransitionAnimator = FullScreenImageTransitionAnimator()\n\n    public private(set) lazy var singleTapGestureRecognizer: UITapGestureRecognizer = {\n        return UITapGestureRecognizer(target: self, action: #selector(FullScreenImageBrowser.handleSingleTapGestureRecognizer(_:)))\n    }()\n    public private(set) lazy var panGestureRecognizer: UIPanGestureRecognizer = {\n        return UIPanGestureRecognizer(target: self, action: #selector(FullScreenImageBrowser.handlePanGestureRecognizer(_:)))\n    }()\n    \n    /*\n     * The mask view displayed over images\n     */\n    public var maskView: MaskImageView = MaskImageView(frame: .zero) {\n        willSet {\n            maskView.removeFromSuperview()\n        }\n        didSet {\n            maskView.imagesBrowser = self\n            maskView.autoresizingMask = [.flexibleWidth, .flexibleHeight]\n            maskView.frame = view.bounds\n            view.addSubview(maskView)\n        }\n    }\n\n    public var currentMedia: MediaDownloadable? {\n        return currentImageViewer?.media\n    }\n\n    private var statusBarHidden = false\n\n    // MARK: - Init\n    required public init?(coder aDecoder: NSCoder) {\n        viewModel = FullScreenImageBrowserViewModel(media: [])\n        pageViewController = UIPageViewController()\n        super.init(nibName: nil, bundle: nil)\n        initialSetupWithImage(nil)\n    }\n\n    public override init(nibName nibNameOrNil: String!, bundle nibBundleOrNil: Bundle!) {\n        viewModel = FullScreenImageBrowserViewModel(media: [])\n        pageViewController = UIPageViewController()\n        super.init(nibName: nil, bundle: nil)\n        initialSetupWithImage(nil)\n    }\n\n    /**\n     The designated initializer\n\n     - parameter viewModel:     View model instance for Full Screen Image Browser.\n     - parameter startingImage: The image to be displayed at first place when launching image browser.\n     - parameter referenceView: The view from which to animate.\n\n     - returns: an instance of full screen image browser\n     */\n    public required init(viewModel: FullScreenImageBrowserViewModel,\n                startingImage: MediaDownloadable? = nil,\n                referenceView: UIView? = nil) {\n        self.viewModel = viewModel\n        pageViewController = UIPageViewController()\n        super.init(nibName: nil, bundle: nil)\n\n        initialSetupWithImage(startingImage == nil ? viewModel.media.first : startingImage)\n        transitionAnimator.startingView = referenceView\n        transitionAnimator.endingView = currentImageViewer?.zoomableImageview.imageView\n    }\n\n    private func initialSetupWithImage(_ image: MediaDownloadable? = nil) {\n        maskView.imagesBrowser = self\n        setupPageViewControllerWith(image)\n\n        modalPresentationStyle = .custom\n        transitioningDelegate = self\n        modalPresentationCapturesStatusBarAppearance = true\n\n        let textColor = view.tintColor ?? UIColor.white\n        #if swift(>=4.0)\n            maskView.titleTextAttributes = [NSAttributedStringKey.foregroundColor: textColor]\n        #else\n            maskView.titleTextAttributes = [NSForegroundColorAttributeName: textColor]\n        #endif\n    }\n\n    private func setupPageViewControllerWith(_ image: MediaDownloadable? = nil) {\n        pageViewController = UIPageViewController(transitionStyle: .scroll, navigationOrientation: .horizontal, options: [UIPageViewControllerOptionInterPageSpacingKey: 16.0])\n        pageViewController.view.backgroundColor = .clear\n        pageViewController.delegate = self\n        pageViewController.dataSource = self\n\n        if let _image = image, viewModel.containsMedia(_image) {\n            changeToImage(_image, animated: false)\n        } else if let _image = viewModel.media.first {\n            changeToImage(_image, animated: false)\n        }\n    }\n\n    private func setupMaskView() {\n        maskView.autoresizingMask = [.flexibleWidth, .flexibleHeight]\n        maskView.frame = view.bounds\n        view.addSubview(maskView)\n        maskView.setHidden(true, animated: false)\n    }\n\n    deinit {\n        pageViewController.delegate = nil\n        pageViewController.dataSource = nil\n    }\n\n    // MARK: - View Controller Life Cycle\n    override public func viewDidLoad() {\n        super.viewDidLoad()\n        view.tintColor = .white\n        view.backgroundColor = .black\n        pageViewController.view.backgroundColor = .clear\n\n        pageViewController.view.addGestureRecognizer(panGestureRecognizer)\n        pageViewController.view.addGestureRecognizer(singleTapGestureRecognizer)\n\n        addChildViewController(pageViewController)\n        view.addSubview(pageViewController.view)\n        pageViewController.view.autoresizingMask = [.flexibleWidth, .flexibleHeight]\n        pageViewController.didMove(toParentViewController: self)\n\n        setupMaskView()\n    }\n\n    override public func viewDidAppear(_ animated: Bool) {\n        super.viewDidAppear(animated)\n\n        // This fix issue that navigationBar animate to up\n        // when presentingViewController is UINavigationViewController\n        statusBarHidden = true\n        UIView.animate(withDuration: 0.25) { self.setNeedsStatusBarAppearanceUpdate() }\n        updateCurrentImageInfo()\n    }\n\n    // MARK: - Public\n\n    /**\n     Displays the specified image. Can be called before the view controller is displayed.\n\n     - parameter media:    The photo to make the currently displayed photo.\n     - parameter animated: Whether to animate the transition to the new photo.\n     */\n    public func changeToImage(_ media: MediaDownloadable,\n                              animated: Bool,\n                              direction: UIPageViewControllerNavigationDirection = .forward) {\n        if !viewModel.containsMedia(media) { return }\n\n        let imageViewer = SingleMediaViewerFor(media)\n        pageViewController.setViewControllers([imageViewer], direction: direction, animated: animated, completion: nil)\n        updateCurrentImageInfo()\n    }\n\n    private func updateCurrentImageInfo() {\n        if let _currentImage = currentMedia {\n            maskView.populateWithImage(_currentImage)\n        }\n    }\n}\n\n// MARK: - UIPageViewController DataSource Delegate\nextension FullScreenImageBrowser: UIPageViewControllerDataSource, UIPageViewControllerDelegate {\n\n    /*\n     * Currently displayed by page view controller\n     */\n    public var currentImageViewer: SingleMediaViewer? {\n        return pageViewController.viewControllers?.first as? SingleMediaViewer\n    }\n\n    public func SingleMediaViewerFor(_ media: MediaDownloadable) -> SingleMediaViewer {\n        let imageViewer = SingleMediaViewer(media: media)\n        singleTapGestureRecognizer.require(toFail: imageViewer.doubleTapGestureRecognizer)\n\n        return imageViewer\n    }\n\n    @objc public func pageViewController(_ pageViewController: UIPageViewController, viewControllerBefore viewController: UIViewController) -> UIViewController? {\n        guard let imageViewer = viewController as? SingleMediaViewer,\n            let index = viewModel.indexOfMedia(imageViewer.media),\n            let newImage = viewModel.mediaAtIndex(index - 1) else {\n                return nil\n        }\n        return SingleMediaViewerFor(newImage)\n    }\n\n    @objc public func pageViewController(_ pageViewController: UIPageViewController, viewControllerAfter viewController: UIViewController) -> UIViewController? {\n        guard let imageViewer = viewController as? SingleMediaViewer,\n            let index = viewModel.indexOfMedia(imageViewer.media),\n            let newImage = viewModel.mediaAtIndex(index + 1) else {\n                return nil\n        }\n        return SingleMediaViewerFor(newImage)\n    }\n\n    @objc public func pageViewController(_ pageViewController: UIPageViewController, didFinishAnimating finished: Bool, previousViewControllers: [UIViewController], transitionCompleted completed: Bool) {\n        if completed {\n            updateCurrentImageInfo()\n        }\n    }\n}\n\n// MARK: - UIViewController Transitioning Delegate\nextension FullScreenImageBrowser: UIViewControllerTransitioningDelegate {\n    public func animationController(forPresented presented: UIViewController, presenting: UIViewController, source: UIViewController) -> UIViewControllerAnimatedTransitioning? {\n        transitionAnimator.dismissing = false\n        return transitionAnimator\n    }\n\n    public func animationController(forDismissed dismissed: UIViewController) -> UIViewControllerAnimatedTransitioning? {\n        transitionAnimator.dismissing = true\n        return transitionAnimator\n    }\n}\n\n// MARK: - Gesture Recognizer\nextension FullScreenImageBrowser {\n    @objc private func handleSingleTapGestureRecognizer(_ gestureRecognizer: UITapGestureRecognizer) {\n        maskView.setHidden(!maskView.isHidden, animated: true)\n\n        guard let currentMedia = currentMedia, currentMedia.isVideoThumbnail == true else { return }\n        playVideo()\n    }\n\n    @objc private func handlePanGestureRecognizer(_ gestureRecognizer: UIPanGestureRecognizer) {\n        dismiss(animated: true, completion: nil)\n    }\n}\n\n// MARK: - Status Bar\nextension FullScreenImageBrowser {\n    public override var prefersStatusBarHidden: Bool {\n        if let parentStatusBarHidden = presentingViewController?.prefersStatusBarHidden , parentStatusBarHidden == true {\n            return parentStatusBarHidden\n        }\n        return statusBarHidden\n    }\n\n    public override var preferredStatusBarStyle: UIStatusBarStyle {\n        return .default\n    }\n\n    public override var preferredStatusBarUpdateAnimation: UIStatusBarAnimation {\n        return .fade\n    }\n}\n\n// MARK: - Video Showcase\nextension FullScreenImageBrowser {\n    public func playVideo() {\n        guard let currentMedia = currentMedia else { return }\n        guard let videoURL = currentMedia.videoURL else { debugPrint(\"\\(#file) invalid url found for video\"); return }\n        let player = AVPlayer(url: videoURL)\n        let playerController = AVPlayerViewController()\n        playerController.player = player\n        present(playerController, animated: true, completion: nil)\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/core/FullScreenImageBrowserViewModel.swift",
    "content": "//\n//  FullScreenImageBrowserViewModel.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Alex Jiang on 26/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport Foundation\n\npublic struct FullScreenImageBrowserViewModel {\n\n    /// Designated Init\n    ///\n    /// - Parameter media: a collection of media to be either video or image\n    public init(media: [MediaDownloadable]) {\n        self.media = media\n    }\n\n    public private(set) var media: [MediaDownloadable]\n\n    // MARK: - Media\n\n    public var numberOfImages: Int {\n        return media.count\n    }\n\n    public func mediaAtIndex(_ index: Int) -> MediaDownloadable? {\n        if (index < media.count && index >= 0) {\n            return media[index]\n        }\n        return nil\n    }\n\n    public func indexOfMedia(_ media: MediaDownloadable) -> Int? {\n        return self.media.index(where: { $0 === media })\n    }\n\n    public func containsMedia(_ media: MediaDownloadable) -> Bool {\n        return indexOfMedia(media) != nil\n    }\n\n    // MARK: - Video\n\n    public var shouldShowVideo: Bool {\n        return media.contains { $0.isVideoThumbnail == true }\n    }\n\n    public func videoURLAtIndex(_ index: Int) -> URL? {\n        guard index < media.count && index > 0 else { return nil }\n        guard let videoURL = media[index].videoURL else { return nil }\n\n        return videoURL\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/core/MaskImageViewer.swift",
    "content": "//\n//  MaskImageViewer.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\npublic protocol MaskImageViewable: class {\n    var imagesBrowser: FullScreenImageBrowser? { get set }\n\n    func populateWithImage(_ image: MediaDownloadable)\n    func setHidden(_ hidden: Bool, animated: Bool)\n}\n\npublic final class MaskImageView: UIView , MaskImageViewable {\n    public private(set) var navigationBar: UINavigationBar!\n\n    public private(set) var navigationItem: UINavigationItem!\n    public weak var imagesBrowser: FullScreenImageBrowser?\n    private var currentMedia: MediaDownloadable?\n\n    public var leftBarButtonItem: UIBarButtonItem? {\n        didSet {\n            navigationItem.leftBarButtonItem = leftBarButtonItem\n        }\n    }\n\n    #if swift(>=4.0)\n    public var titleTextAttributes: [NSAttributedStringKey : AnyObject] = [:] {\n        didSet {\n            navigationBar.titleTextAttributes = titleTextAttributes\n        }\n    }\n    #else\n    public var titleTextAttributes: [String : AnyObject] = [:] {\n        didSet {\n            navigationBar.titleTextAttributes = titleTextAttributes\n        }\n    }\n    #endif\n\n    public override init(frame: CGRect) {\n        super.init(frame: frame)\n        setupNavigationBar()\n    }\n\n    required public init?(coder aDecoder: NSCoder) {\n        fatalError(\"init(coder:) has not been implemented\")\n    }\n\n    public override func hitTest(_ point: CGPoint, with event: UIEvent?) -> UIView? {\n        if let hitView = super.hitTest(point, with: event) , hitView != self {\n            return hitView\n        }\n        return nil\n    }\n\n    public override func layoutSubviews() {\n        UIView.performWithoutAnimation {\n            self.navigationBar.invalidateIntrinsicContentSize()\n            self.navigationBar.layoutIfNeeded()\n        }\n        super.layoutSubviews()\n    }\n\n    public func setHidden(_ hidden: Bool, animated: Bool) {\n        if isHidden == hidden { return }\n        if !animated { isHidden = hidden; return }\n\n        isHidden = false\n        alpha = hidden ? 1.0 : 0.0\n\n        UIView.animate(withDuration: 0.2,\n                       delay: 0.0,\n                       options: [.allowAnimatedContent, .allowUserInteraction],\n                       animations: { self.alpha = hidden ? 0.0 : 1.0 },\n                       completion: { _ in self.alpha = 1.0; self.isHidden = hidden })\n    }\n\n    public func populateWithImage(_ media: MediaDownloadable) {\n        currentMedia = media\n\n        guard let _imagesBrowser = imagesBrowser,\n              let index = imagesBrowser?.viewModel.indexOfMedia(media) else { return }\n\n        navigationItem.title = String(format:NSLocalizedString(\"%d of %d\",comment:\"\"),\n                                      index+1,\n                                      _imagesBrowser.viewModel.numberOfImages)\n    }\n\n    @objc private func closeButtonTapped(_ sender: UIBarButtonItem) {\n        imagesBrowser?.dismiss(animated: true, completion: nil)\n    }\n\n    private func setupNavigationBar() {\n        navigationBar = UINavigationBar()\n        navigationBar.translatesAutoresizingMaskIntoConstraints = false\n        navigationBar.backgroundColor = UIColor.clear\n        navigationBar.barTintColor = nil\n        navigationBar.isTranslucent = true\n        navigationBar.shadowImage = UIImage()\n        navigationBar.setBackgroundImage(UIImage(), for: .default)\n\n        navigationItem = UINavigationItem(title: \"\")\n        navigationBar.items = [navigationItem]\n        addSubview(navigationBar)\n\n        let topConstraint: NSLayoutConstraint\n        if #available(iOS 11.0, *) {\n            topConstraint = navigationBar.topAnchor.constraint(equalTo: safeAreaLayoutGuide.topAnchor)\n        } else {\n            topConstraint = navigationBar.topAnchor.constraint(equalTo: self.topAnchor)\n        }\n        let widthConstraint = navigationBar.widthAnchor.constraint(equalTo: self.widthAnchor)\n        let horizontalConstraint = navigationBar.centerXAnchor.constraint(equalTo: self.centerXAnchor)\n        NSLayoutConstraint.activate([topConstraint, widthConstraint, horizontalConstraint])\n\n        if let bundlePath = Bundle(for: type(of: self)).path(forResource: \"FullScreenImageBrowser\", ofType: \"bundle\") {\n            let bundle = Bundle(path: bundlePath)\n            leftBarButtonItem = UIBarButtonItem(image: UIImage(named: \"close\", in: bundle, compatibleWith: nil),\n                                                style: .plain,\n                                                target: self,\n                                                action: #selector(MaskImageView.closeButtonTapped(_:)))\n        } else {\n            leftBarButtonItem = UIBarButtonItem(barButtonSystemItem: .cancel,\n                                                target: self,\n                                                action: #selector(MaskImageView.closeButtonTapped(_:)))\n        }\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/core/MediaDownloadable.swift",
    "content": "//\n//  MediaDownloadable.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\n/// Media Type protocol for defining media to be either image, or video\npublic protocol MediaDownloadable: class {\n    var image: UIImage? { get }\n    var imageURL: URL? { get }\n    var isVideoThumbnail: Bool { get set }\n    var videoURL: URL? { get }\n    \n    func loadImageWithCompletionHandler(_ completion: @escaping (_ image: UIImage?, _ error: NSError?) -> ())\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/core/SingleImageViewer.swift",
    "content": "//\n//  SingleMediaViewer.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\npublic final class SingleMediaViewer: UIViewController, UIScrollViewDelegate {\n\n    // MARK: - Property\n    public var media: MediaDownloadable\n\n    public lazy private(set) var zoomableImageview: ZoomableImageView = {\n        return ZoomableImageView()\n    }()\n\n    public lazy private(set) var doubleTapGestureRecognizer: UITapGestureRecognizer = {\n        let gesture = UITapGestureRecognizer(target: self, action: #selector(SingleMediaViewer.handleDoubleTapWithGestureRecognizer(_:)))\n        gesture.numberOfTapsRequired = 2\n        return gesture\n    }()\n\n    public lazy private(set) var activityIndicator: UIActivityIndicatorView = {\n        let activityIndicator = UIActivityIndicatorView(activityIndicatorStyle: .white)\n        activityIndicator.startAnimating()\n        return activityIndicator\n    }()\n\n    // MARK: - Init\n    public init(media: MediaDownloadable) {\n        self.media = media\n        super.init(nibName: nil, bundle: nil)\n    }\n\n    required public init?(coder aDecoder: NSCoder) {\n        fatalError(\"init(coder:) has not been implemented\")\n    }\n\n    deinit {\n        zoomableImageview.delegate = nil\n    }\n\n    //  MARK: - View Controller Life Cycle\n    public override func viewDidLoad() {\n        super.viewDidLoad()\n\n        zoomableImageview.delegate = self\n        zoomableImageview.frame = view.bounds\n        zoomableImageview.autoresizingMask = [.flexibleWidth, .flexibleHeight]\n        view.addSubview(zoomableImageview)\n\n        view.addSubview(activityIndicator)\n        activityIndicator.center = CGPoint(x: view.bounds.midX, y: view.bounds.midY)\n        activityIndicator.autoresizingMask = [.flexibleTopMargin, .flexibleLeftMargin, .flexibleRightMargin, .flexibleBottomMargin]\n        activityIndicator.sizeToFit()\n\n        view.addGestureRecognizer(doubleTapGestureRecognizer)\n\n        if let image = media.image {\n            zoomableImageview.image = image\n            activityIndicator.stopAnimating()\n        } else {\n            loadmedia()\n        }\n    }\n\n    public override func viewWillLayoutSubviews() {\n        super.viewWillLayoutSubviews()\n        zoomableImageview.frame = view.bounds\n    }\n\n    // MARK: - Private\n    private func loadmedia() {\n        view.bringSubview(toFront: activityIndicator)\n        media.loadImageWithCompletionHandler({ [weak self] (image, error) -> () in\n            let completeLoading = {\n                self?.activityIndicator.stopAnimating()\n                self?.zoomableImageview.image = image\n            }\n\n            if Thread.isMainThread {\n                completeLoading()\n            } else {\n                DispatchQueue.main.async(execute: { () -> Void in\n                    completeLoading()\n                })\n            }\n        })\n    }\n\n    @objc private func handleDoubleTapWithGestureRecognizer(_ recognizer: UITapGestureRecognizer) {\n        let pointInView = recognizer.location(in: zoomableImageview.imageView)\n        var newZoomScale = zoomableImageview.maximumZoomScale\n\n        if zoomableImageview.zoomScale >= zoomableImageview.maximumZoomScale ||\n            abs(zoomableImageview.zoomScale - zoomableImageview.maximumZoomScale) <= 0.01 {\n            newZoomScale = zoomableImageview.minimumZoomScale\n        }\n\n        let scrollViewSize = zoomableImageview.bounds.size\n        let width = scrollViewSize.width / newZoomScale\n        let height = scrollViewSize.height / newZoomScale\n        let originX = pointInView.x - (width / 2.0)\n        let originY = pointInView.y - (height / 2.0)\n\n        let rectToZoom = CGRect(x: originX, y: originY, width: width, height: height)\n        zoomableImageview.zoom(to: rectToZoom, animated: true)\n    }\n\n    // MARK:- UIScrollViewDelegate\n    public func viewForZooming(in scrollView: UIScrollView) -> UIView? {\n        return zoomableImageview.imageView\n    }\n\n    public func scrollViewWillBeginZooming(_ scrollView: UIScrollView, with view: UIView?) {\n        scrollView.panGestureRecognizer.isEnabled = true\n    }\n\n    public func scrollViewDidEndZooming(_ scrollView: UIScrollView, with view: UIView?, atScale scale: CGFloat) {\n        if (scrollView.zoomScale == scrollView.minimumZoomScale) {\n            scrollView.panGestureRecognizer.isEnabled = false\n        }\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/core/ZoomableImageView.swift",
    "content": "//\n//  ZoomableImageView.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\nfinal public class ZoomableImageView: UIScrollView {\n\n    // MARK: - Property\n    public lazy var imageView: UIImageView = {\n        let imageView = UIImageView(frame: bounds)\n        addSubview(imageView)\n        return imageView\n    }()\n\n    public var image: UIImage? {\n        didSet {\n            updateImage(image)\n        }\n    }\n\n    override public var frame: CGRect {\n        didSet {\n            updateZoomScale()\n            centerScrollViewContents()\n        }\n    }\n\n    // MARK: - Init\n    override public init(frame: CGRect) {\n        super.init(frame: frame)\n        setupImageScrollView()\n        updateZoomScale()\n    }\n\n    required public init?(coder aDecoder: NSCoder) {\n        super.init(coder: aDecoder)\n        setupImageScrollView()\n        updateZoomScale()\n    }\n\n    private func setupImageScrollView() {\n        showsVerticalScrollIndicator = false\n        showsHorizontalScrollIndicator = false;\n        bouncesZoom = true;\n        decelerationRate = UIScrollViewDecelerationRateFast;\n    }\n\n    // MARK: - View Life Cycle\n    override public func didAddSubview(_ subview: UIView) {\n        super.didAddSubview(subview)\n        centerScrollViewContents()\n    }\n\n    // MARK: - Private\n    private func centerScrollViewContents() {\n        var horizontalInset: CGFloat = 0\n        var verticalInset: CGFloat = 0\n\n        if contentSize.width < bounds.width {\n            horizontalInset = (bounds.width - contentSize.width) * 0.5\n        }\n\n        if contentSize.height < bounds.height {\n            verticalInset = (bounds.height - contentSize.height) * 0.5\n        }\n\n        if window?.screen.scale < 2.0 {\n            horizontalInset = floor(horizontalInset)\n            verticalInset = floor(verticalInset)\n        }\n\n        contentInset = UIEdgeInsetsMake(verticalInset, horizontalInset, verticalInset, horizontalInset);\n    }\n\n    private func updateImage(_ image: UIImage?) {\n        let size = image?.size ?? CGSize.zero\n\n        imageView.transform = CGAffineTransform.identity\n        imageView.image = image\n        imageView.frame = CGRect(origin: .zero, size: size)\n        contentSize = size\n\n        updateZoomScale()\n        centerScrollViewContents()\n    }\n\n    private func updateZoomScale() {\n        guard let image = imageView.image else { return }\n\n        let scrollViewFrame = bounds\n        let scaleWidth = scrollViewFrame.size.width / image.size.width\n        let scaleHeight = scrollViewFrame.size.height / image.size.height\n        let minScale = min(scaleWidth, scaleHeight)\n\n        minimumZoomScale = minScale\n        maximumZoomScale = max(minScale, maximumZoomScale)\n\n        if abs(minScale - maximumZoomScale) < 0.01 {\n            maximumZoomScale = minScale * 3.0\n        }\n\n        zoomScale = minimumZoomScale\n        panGestureRecognizer.isEnabled = false\n    }\n}\n\n\n/// Make binary `<` operator to accept optional\n///\n/// - Parameters:\n///   - lhs: expression on the left hand side of the `<`\n///   - rhs: expression on the right hand side of the `<`\n/// - Returns: Boolen value of the comparsion\nprivate func < <T : Comparable>(lhs: T?, rhs: T?) -> Bool {\n    switch (lhs, rhs) {\n    case let (l?, r?):\n        return l < r\n    case (nil, _?):\n        return true\n    default:\n        return false\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/helper/SingleMedia.swift",
    "content": "//\n//  SingleMedia.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Alex Jiang on 26/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\n/// A concrete media type for supporting both image and video to be downloadable\npublic final class SingleMedia: MediaDownloadable {\n\n    // MARK: - Properties\n    public var image: UIImage?\n    public var imageURL: URL?\n    public var videoURL: URL?\n    public var isVideoThumbnail: Bool\n\n    // MARK: - Init\n    public init(imageURL: URL?, isVideoThumbnail: Bool = false, videoURL: URL? = nil) {\n        self.imageURL = imageURL\n        self.videoURL = videoURL\n        self.isVideoThumbnail = isVideoThumbnail\n    }\n\n    // MARK: - Downloadable\n    public func loadImageWithCompletionHandler(_ completion: @escaping (UIImage?, NSError?) -> ()) {\n        if let image = image {\n            completion(image, nil)\n            return\n        }\n        loadImageWithURL(imageURL, completion: completion)\n    }\n\n    // override this method to use your favourite networking service\n    public func loadImageWithURL(_ url: URL?, completion: @escaping (_ image: UIImage?, _ error: NSError?) -> ()) {\n        let session = URLSession(configuration: URLSessionConfiguration.default)\n        guard let imageURL = url else { completion(nil, NSError(domain: \"FullScreenImageBrowserDomain\", code: -2, userInfo: [ NSLocalizedDescriptionKey: \"Image URL not found.\"])); return }\n\n        session.dataTask(with: imageURL, completionHandler: {[unowned self] (response, data, error) in\n            DispatchQueue.main.async {\n                if error != nil {\n                    completion(nil, error as NSError?)\n                } else if let response = response, let image = UIImage(data: response) {\n                    completion(self.isVideoThumbnail ? image.aj_imageWithPlayIcon() : image, nil)\n                } else {\n                    completion(nil, NSError(domain: \"FullScreenImageBrowserDomain\", code: -1, userInfo: [ NSLocalizedDescriptionKey: \"Couldn't load image\"]))\n                }\n                session.finishTasksAndInvalidate()\n            }\n        }).resume()\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/helper/UIImage+Ex.swift",
    "content": "//\n//  UIImage+Ex.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Alex Jiang on 27/4/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\npublic extension UIImage {\n    public func aj_imageWithPlayIcon() -> UIImage {\n        guard let bundlePath = Bundle(for: SingleMedia.self).path(forResource: \"FullScreenImageBrowser\", ofType: \"bundle\") else { return self }\n        guard let playButtonImage = UIImage(named: \"video-play-icon\", in: Bundle(path: bundlePath), compatibleWith: nil) else { return self }\n\n        UIGraphicsBeginImageContextWithOptions(size, false, 0.0)\n        draw(in: CGRect(x: 0, y: 0, width: size.width, height: size.height))\n        let width = min(size.width, size.height) * 0.2\n        let x = size.width / 2.0 - width / 2.0\n        let y = size.height / 2.0 - width / 2.0\n        playButtonImage.draw(in: CGRect(x: x, y: y, width: width, height: width))\n\n        guard let result = UIGraphicsGetImageFromCurrentImageContext() else { return self }\n        UIGraphicsEndImageContext()\n\n        return result\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser/helper/UIView+SnapShot.swift",
    "content": "//\n//  UIView+SnapShot.swift\n//  A_J_Full_Screen_Image_Browser\n//\n//  Created by Junliang Jiang on 25/2/18.\n//  Copyright © 2018 Junliang Jiang. All rights reserved.\n//\n\nimport UIKit\n\npublic extension UIView {\n\n    /// Create snapshot view with layer transform information if available\n    ///\n    /// - Returns: A new view object based on a snapshot of the current view’s rendered contents\n    public func aj_snapshotView() -> UIView {\n        guard let contents = layer.contents else {\n            return snapshotView(afterScreenUpdates: true) ?? UIView()\n        }\n\n        var snapshotedView: UIView!\n\n        if let view = self as? UIImageView {\n            snapshotedView = type(of: view).init(image: view.image)\n            snapshotedView.bounds = view.bounds\n        } else {\n            snapshotedView = UIView(frame: frame)\n            snapshotedView.layer.contents = contents\n            snapshotedView.layer.bounds = layer.bounds\n        }\n        snapshotedView.layer.cornerRadius = layer.cornerRadius\n        snapshotedView.layer.masksToBounds = layer.masksToBounds\n        snapshotedView.contentMode = contentMode\n        snapshotedView.transform = transform\n\n        return snapshotedView\n    }\n\n    /// Converts a point from the coordinate space of the current object to the container view coordinate space.\n    ///\n    /// - Parameter containerView: container view for the point\n    /// - Returns: A point specified in the container view coordinate space.\n    public func aj_translatedCenterPointToContainerView(_ containerView: UIView) -> CGPoint {\n        var centerPoint = center\n\n        if let scrollView = self.superview as? UIScrollView , scrollView.zoomScale != 1.0 {\n            centerPoint.x += (scrollView.bounds.width - scrollView.contentSize.width) / 2.0 + scrollView.contentOffset.x\n            centerPoint.y += (scrollView.bounds.height - scrollView.contentSize.height) / 2.0 + scrollView.contentOffset.y\n        }\n        return self.superview?.convert(centerPoint, to: containerView) ?? .zero\n    }\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser.xcodeproj/project.pbxproj",
    "content": "// !$*UTF8*$!\n{\n\tarchiveVersion = 1;\n\tclasses = {\n\t};\n\tobjectVersion = 48;\n\tobjects = {\n\n/* Begin PBXBuildFile section */\n\t\t14DE4D4A20424E77003C7BF4 /* AppDelegate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 14DE4D4920424E77003C7BF4 /* AppDelegate.swift */; };\n\t\t14DE4D4C20424E77003C7BF4 /* ViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 14DE4D4B20424E77003C7BF4 /* ViewController.swift */; };\n\t\t14DE4D4F20424E77003C7BF4 /* Main.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 14DE4D4D20424E77003C7BF4 /* Main.storyboard */; };\n\t\t14DE4D5120424E77003C7BF4 /* Assets.xcassets in Resources */ = {isa = PBXBuildFile; fileRef = 14DE4D5020424E77003C7BF4 /* Assets.xcassets */; };\n\t\t14DE4D5420424E77003C7BF4 /* LaunchScreen.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 14DE4D5220424E77003C7BF4 /* LaunchScreen.storyboard */; };\n\t\tD6E787282092CEC1007E64C1 /* UIImage+Ex.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6E787272092CEC1007E64C1 /* UIImage+Ex.swift */; };\n\t\tD6EE23322044E41B00E8C3AF /* SingleMedia.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE23302044E41B00E8C3AF /* SingleMedia.swift */; };\n\t\tD6EE23332044E41B00E8C3AF /* UIView+SnapShot.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE23312044E41B00E8C3AF /* UIView+SnapShot.swift */; };\n\t\tD6EE233F2044E45700E8C3AF /* FullScreenImageBrowser.bundle in Resources */ = {isa = PBXBuildFile; fileRef = D6EE23352044E45700E8C3AF /* FullScreenImageBrowser.bundle */; };\n\t\tD6EE23402044E45700E8C3AF /* FullScreenImageTransitionAnimator.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE23372044E45700E8C3AF /* FullScreenImageTransitionAnimator.swift */; };\n\t\tD6EE23412044E45700E8C3AF /* FullScreenImageBrowser.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE23392044E45700E8C3AF /* FullScreenImageBrowser.swift */; };\n\t\tD6EE23422044E45700E8C3AF /* FullScreenImageBrowserViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE233A2044E45700E8C3AF /* FullScreenImageBrowserViewModel.swift */; };\n\t\tD6EE23432044E45700E8C3AF /* MediaDownloadable.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE233B2044E45700E8C3AF /* MediaDownloadable.swift */; };\n\t\tD6EE23442044E45700E8C3AF /* MaskImageViewer.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE233C2044E45700E8C3AF /* MaskImageViewer.swift */; };\n\t\tD6EE23452044E45700E8C3AF /* SingleImageViewer.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE233D2044E45700E8C3AF /* SingleImageViewer.swift */; };\n\t\tD6EE23462044E45700E8C3AF /* ZoomableImageView.swift in Sources */ = {isa = PBXBuildFile; fileRef = D6EE233E2044E45700E8C3AF /* ZoomableImageView.swift */; };\n/* End PBXBuildFile section */\n\n/* Begin PBXFileReference section */\n\t\t14DE4D4620424E77003C7BF4 /* A_J_Full_Screen_Image_Browser.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = A_J_Full_Screen_Image_Browser.app; sourceTree = BUILT_PRODUCTS_DIR; };\n\t\t14DE4D4920424E77003C7BF4 /* AppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AppDelegate.swift; sourceTree = \"<group>\"; };\n\t\t14DE4D4B20424E77003C7BF4 /* ViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ViewController.swift; sourceTree = \"<group>\"; };\n\t\t14DE4D4E20424E77003C7BF4 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/Main.storyboard; sourceTree = \"<group>\"; };\n\t\t14DE4D5020424E77003C7BF4 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = \"<group>\"; };\n\t\t14DE4D5320424E77003C7BF4 /* Base */ = {isa = PBXFileReference; lastKnownFileType = file.storyboard; name = Base; path = Base.lproj/LaunchScreen.storyboard; sourceTree = \"<group>\"; };\n\t\t14DE4D5520424E77003C7BF4 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = \"<group>\"; };\n\t\tD6E787272092CEC1007E64C1 /* UIImage+Ex.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = \"UIImage+Ex.swift\"; sourceTree = \"<group>\"; };\n\t\tD6EE23302044E41B00E8C3AF /* SingleMedia.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SingleMedia.swift; sourceTree = \"<group>\"; };\n\t\tD6EE23312044E41B00E8C3AF /* UIView+SnapShot.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = \"UIView+SnapShot.swift\"; sourceTree = \"<group>\"; };\n\t\tD6EE23352044E45700E8C3AF /* FullScreenImageBrowser.bundle */ = {isa = PBXFileReference; lastKnownFileType = \"wrapper.plug-in\"; path = FullScreenImageBrowser.bundle; sourceTree = \"<group>\"; };\n\t\tD6EE23372044E45700E8C3AF /* FullScreenImageTransitionAnimator.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = FullScreenImageTransitionAnimator.swift; sourceTree = \"<group>\"; };\n\t\tD6EE23392044E45700E8C3AF /* FullScreenImageBrowser.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = FullScreenImageBrowser.swift; sourceTree = \"<group>\"; };\n\t\tD6EE233A2044E45700E8C3AF /* FullScreenImageBrowserViewModel.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = FullScreenImageBrowserViewModel.swift; sourceTree = \"<group>\"; };\n\t\tD6EE233B2044E45700E8C3AF /* MediaDownloadable.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MediaDownloadable.swift; sourceTree = \"<group>\"; };\n\t\tD6EE233C2044E45700E8C3AF /* MaskImageViewer.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = MaskImageViewer.swift; sourceTree = \"<group>\"; };\n\t\tD6EE233D2044E45700E8C3AF /* SingleImageViewer.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SingleImageViewer.swift; sourceTree = \"<group>\"; };\n\t\tD6EE233E2044E45700E8C3AF /* ZoomableImageView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ZoomableImageView.swift; sourceTree = \"<group>\"; };\n/* End PBXFileReference section */\n\n/* Begin PBXFrameworksBuildPhase section */\n\t\t14DE4D4320424E77003C7BF4 /* Frameworks */ = {\n\t\t\tisa = PBXFrameworksBuildPhase;\n\t\t\tbuildActionMask = 2147483647;\n\t\t\tfiles = (\n\t\t\t);\n\t\t\trunOnlyForDeploymentPostprocessing = 0;\n\t\t};\n/* End PBXFrameworksBuildPhase section */\n\n/* Begin PBXGroup section */\n\t\t14DE4D3D20424E77003C7BF4 = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\t14DE4D4820424E77003C7BF4 /* A_J_Full_Screen_Image_Browser */,\n\t\t\t\t14DE4D4720424E77003C7BF4 /* Products */,\n\t\t\t);\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\t14DE4D4720424E77003C7BF4 /* Products */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\t14DE4D4620424E77003C7BF4 /* A_J_Full_Screen_Image_Browser.app */,\n\t\t\t);\n\t\t\tname = Products;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\t14DE4D4820424E77003C7BF4 /* A_J_Full_Screen_Image_Browser */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tD6EE23382044E45700E8C3AF /* core */,\n\t\t\t\tD6EE232F2044E41B00E8C3AF /* helper */,\n\t\t\t\tD6EE23342044E45700E8C3AF /* asset */,\n\t\t\t\tD6EE23362044E45700E8C3AF /* animator */,\n\t\t\t\t14DE4D4920424E77003C7BF4 /* AppDelegate.swift */,\n\t\t\t\t14DE4D4B20424E77003C7BF4 /* ViewController.swift */,\n\t\t\t\t14DE4D4D20424E77003C7BF4 /* Main.storyboard */,\n\t\t\t\t14DE4D5020424E77003C7BF4 /* Assets.xcassets */,\n\t\t\t\t14DE4D5220424E77003C7BF4 /* LaunchScreen.storyboard */,\n\t\t\t\t14DE4D5520424E77003C7BF4 /* Info.plist */,\n\t\t\t);\n\t\t\tpath = A_J_Full_Screen_Image_Browser;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tD6EE232F2044E41B00E8C3AF /* helper */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tD6EE23302044E41B00E8C3AF /* SingleMedia.swift */,\n\t\t\t\tD6EE23312044E41B00E8C3AF /* UIView+SnapShot.swift */,\n\t\t\t\tD6E787272092CEC1007E64C1 /* UIImage+Ex.swift */,\n\t\t\t);\n\t\t\tpath = helper;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tD6EE23342044E45700E8C3AF /* asset */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tD6EE23352044E45700E8C3AF /* FullScreenImageBrowser.bundle */,\n\t\t\t);\n\t\t\tpath = asset;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tD6EE23362044E45700E8C3AF /* animator */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tD6EE23372044E45700E8C3AF /* FullScreenImageTransitionAnimator.swift */,\n\t\t\t);\n\t\t\tpath = animator;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\tD6EE23382044E45700E8C3AF /* core */ = {\n\t\t\tisa = PBXGroup;\n\t\t\tchildren = (\n\t\t\t\tD6EE23392044E45700E8C3AF /* FullScreenImageBrowser.swift */,\n\t\t\t\tD6EE233A2044E45700E8C3AF /* FullScreenImageBrowserViewModel.swift */,\n\t\t\t\tD6EE233B2044E45700E8C3AF /* MediaDownloadable.swift */,\n\t\t\t\tD6EE233C2044E45700E8C3AF /* MaskImageViewer.swift */,\n\t\t\t\tD6EE233D2044E45700E8C3AF /* SingleImageViewer.swift */,\n\t\t\t\tD6EE233E2044E45700E8C3AF /* ZoomableImageView.swift */,\n\t\t\t);\n\t\t\tpath = core;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n/* End PBXGroup section */\n\n/* Begin PBXNativeTarget section */\n\t\t14DE4D4520424E77003C7BF4 /* A_J_Full_Screen_Image_Browser */ = {\n\t\t\tisa = PBXNativeTarget;\n\t\t\tbuildConfigurationList = 14DE4D5820424E77003C7BF4 /* Build configuration list for PBXNativeTarget \"A_J_Full_Screen_Image_Browser\" */;\n\t\t\tbuildPhases = (\n\t\t\t\t14DE4D4220424E77003C7BF4 /* Sources */,\n\t\t\t\t14DE4D4320424E77003C7BF4 /* Frameworks */,\n\t\t\t\t14DE4D4420424E77003C7BF4 /* Resources */,\n\t\t\t);\n\t\t\tbuildRules = (\n\t\t\t);\n\t\t\tdependencies = (\n\t\t\t);\n\t\t\tname = A_J_Full_Screen_Image_Browser;\n\t\t\tproductName = A_J_Full_Screen_Image_Browser;\n\t\t\tproductReference = 14DE4D4620424E77003C7BF4 /* A_J_Full_Screen_Image_Browser.app */;\n\t\t\tproductType = \"com.apple.product-type.application\";\n\t\t};\n/* End PBXNativeTarget section */\n\n/* Begin PBXProject section */\n\t\t14DE4D3E20424E77003C7BF4 /* Project object */ = {\n\t\t\tisa = PBXProject;\n\t\t\tattributes = {\n\t\t\t\tLastSwiftUpdateCheck = 0920;\n\t\t\t\tLastUpgradeCheck = 0920;\n\t\t\t\tORGANIZATIONNAME = \"Junliang Jiang\";\n\t\t\t\tTargetAttributes = {\n\t\t\t\t\t14DE4D4520424E77003C7BF4 = {\n\t\t\t\t\t\tCreatedOnToolsVersion = 9.2;\n\t\t\t\t\t\tProvisioningStyle = Manual;\n\t\t\t\t\t};\n\t\t\t\t};\n\t\t\t};\n\t\t\tbuildConfigurationList = 14DE4D4120424E77003C7BF4 /* Build configuration list for PBXProject \"A_J_Full_Screen_Image_Browser\" */;\n\t\t\tcompatibilityVersion = \"Xcode 8.0\";\n\t\t\tdevelopmentRegion = en;\n\t\t\thasScannedForEncodings = 0;\n\t\t\tknownRegions = (\n\t\t\t\ten,\n\t\t\t\tBase,\n\t\t\t);\n\t\t\tmainGroup = 14DE4D3D20424E77003C7BF4;\n\t\t\tproductRefGroup = 14DE4D4720424E77003C7BF4 /* Products */;\n\t\t\tprojectDirPath = \"\";\n\t\t\tprojectRoot = \"\";\n\t\t\ttargets = (\n\t\t\t\t14DE4D4520424E77003C7BF4 /* A_J_Full_Screen_Image_Browser */,\n\t\t\t);\n\t\t};\n/* End PBXProject section */\n\n/* Begin PBXResourcesBuildPhase section */\n\t\t14DE4D4420424E77003C7BF4 /* Resources */ = {\n\t\t\tisa = PBXResourcesBuildPhase;\n\t\t\tbuildActionMask = 2147483647;\n\t\t\tfiles = (\n\t\t\t\t14DE4D5420424E77003C7BF4 /* LaunchScreen.storyboard in Resources */,\n\t\t\t\t14DE4D5120424E77003C7BF4 /* Assets.xcassets in Resources */,\n\t\t\t\t14DE4D4F20424E77003C7BF4 /* Main.storyboard in Resources */,\n\t\t\t\tD6EE233F2044E45700E8C3AF /* FullScreenImageBrowser.bundle in Resources */,\n\t\t\t);\n\t\t\trunOnlyForDeploymentPostprocessing = 0;\n\t\t};\n/* End PBXResourcesBuildPhase section */\n\n/* Begin PBXSourcesBuildPhase section */\n\t\t14DE4D4220424E77003C7BF4 /* Sources */ = {\n\t\t\tisa = PBXSourcesBuildPhase;\n\t\t\tbuildActionMask = 2147483647;\n\t\t\tfiles = (\n\t\t\t\tD6EE23432044E45700E8C3AF /* MediaDownloadable.swift in Sources */,\n\t\t\t\tD6EE23452044E45700E8C3AF /* SingleImageViewer.swift in Sources */,\n\t\t\t\tD6EE23332044E41B00E8C3AF /* UIView+SnapShot.swift in Sources */,\n\t\t\t\t14DE4D4C20424E77003C7BF4 /* ViewController.swift in Sources */,\n\t\t\t\tD6E787282092CEC1007E64C1 /* UIImage+Ex.swift in Sources */,\n\t\t\t\tD6EE23402044E45700E8C3AF /* FullScreenImageTransitionAnimator.swift in Sources */,\n\t\t\t\tD6EE23322044E41B00E8C3AF /* SingleMedia.swift in Sources */,\n\t\t\t\tD6EE23422044E45700E8C3AF /* FullScreenImageBrowserViewModel.swift in Sources */,\n\t\t\t\tD6EE23442044E45700E8C3AF /* MaskImageViewer.swift in Sources */,\n\t\t\t\t14DE4D4A20424E77003C7BF4 /* AppDelegate.swift in Sources */,\n\t\t\t\tD6EE23462044E45700E8C3AF /* ZoomableImageView.swift in Sources */,\n\t\t\t\tD6EE23412044E45700E8C3AF /* FullScreenImageBrowser.swift in Sources */,\n\t\t\t);\n\t\t\trunOnlyForDeploymentPostprocessing = 0;\n\t\t};\n/* End PBXSourcesBuildPhase section */\n\n/* Begin PBXVariantGroup section */\n\t\t14DE4D4D20424E77003C7BF4 /* Main.storyboard */ = {\n\t\t\tisa = PBXVariantGroup;\n\t\t\tchildren = (\n\t\t\t\t14DE4D4E20424E77003C7BF4 /* Base */,\n\t\t\t);\n\t\t\tname = Main.storyboard;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n\t\t14DE4D5220424E77003C7BF4 /* LaunchScreen.storyboard */ = {\n\t\t\tisa = PBXVariantGroup;\n\t\t\tchildren = (\n\t\t\t\t14DE4D5320424E77003C7BF4 /* Base */,\n\t\t\t);\n\t\t\tname = LaunchScreen.storyboard;\n\t\t\tsourceTree = \"<group>\";\n\t\t};\n/* End PBXVariantGroup section */\n\n/* Begin XCBuildConfiguration section */\n\t\t14DE4D5620424E77003C7BF4 /* Debug */ = {\n\t\t\tisa = XCBuildConfiguration;\n\t\t\tbuildSettings = {\n\t\t\t\tALWAYS_SEARCH_USER_PATHS = NO;\n\t\t\t\tCLANG_ANALYZER_NONNULL = YES;\n\t\t\t\tCLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;\n\t\t\t\tCLANG_CXX_LANGUAGE_STANDARD = \"gnu++14\";\n\t\t\t\tCLANG_CXX_LIBRARY = \"libc++\";\n\t\t\t\tCLANG_ENABLE_MODULES = YES;\n\t\t\t\tCLANG_ENABLE_OBJC_ARC = YES;\n\t\t\t\tCLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;\n\t\t\t\tCLANG_WARN_BOOL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_COMMA = YES;\n\t\t\t\tCLANG_WARN_CONSTANT_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;\n\t\t\t\tCLANG_WARN_DOCUMENTATION_COMMENTS = YES;\n\t\t\t\tCLANG_WARN_EMPTY_BODY = YES;\n\t\t\t\tCLANG_WARN_ENUM_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_INFINITE_RECURSION = YES;\n\t\t\t\tCLANG_WARN_INT_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_OBJC_LITERAL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;\n\t\t\t\tCLANG_WARN_RANGE_LOOP_ANALYSIS = YES;\n\t\t\t\tCLANG_WARN_STRICT_PROTOTYPES = YES;\n\t\t\t\tCLANG_WARN_SUSPICIOUS_MOVE = YES;\n\t\t\t\tCLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;\n\t\t\t\tCLANG_WARN_UNREACHABLE_CODE = YES;\n\t\t\t\tCLANG_WARN__DUPLICATE_METHOD_MATCH = YES;\n\t\t\t\tCODE_SIGN_IDENTITY = \"iPhone Developer\";\n\t\t\t\tCOPY_PHASE_STRIP = NO;\n\t\t\t\tDEBUG_INFORMATION_FORMAT = dwarf;\n\t\t\t\tENABLE_STRICT_OBJC_MSGSEND = YES;\n\t\t\t\tENABLE_TESTABILITY = YES;\n\t\t\t\tGCC_C_LANGUAGE_STANDARD = gnu11;\n\t\t\t\tGCC_DYNAMIC_NO_PIC = NO;\n\t\t\t\tGCC_NO_COMMON_BLOCKS = YES;\n\t\t\t\tGCC_OPTIMIZATION_LEVEL = 0;\n\t\t\t\tGCC_PREPROCESSOR_DEFINITIONS = (\n\t\t\t\t\t\"DEBUG=1\",\n\t\t\t\t\t\"$(inherited)\",\n\t\t\t\t);\n\t\t\t\tGCC_WARN_64_TO_32_BIT_CONVERSION = YES;\n\t\t\t\tGCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;\n\t\t\t\tGCC_WARN_UNDECLARED_SELECTOR = YES;\n\t\t\t\tGCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;\n\t\t\t\tGCC_WARN_UNUSED_FUNCTION = YES;\n\t\t\t\tGCC_WARN_UNUSED_VARIABLE = YES;\n\t\t\t\tIPHONEOS_DEPLOYMENT_TARGET = 11.2;\n\t\t\t\tMTL_ENABLE_DEBUG_INFO = YES;\n\t\t\t\tONLY_ACTIVE_ARCH = YES;\n\t\t\t\tSDKROOT = iphoneos;\n\t\t\t\tSWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG;\n\t\t\t\tSWIFT_OPTIMIZATION_LEVEL = \"-Onone\";\n\t\t\t};\n\t\t\tname = Debug;\n\t\t};\n\t\t14DE4D5720424E77003C7BF4 /* Release */ = {\n\t\t\tisa = XCBuildConfiguration;\n\t\t\tbuildSettings = {\n\t\t\t\tALWAYS_SEARCH_USER_PATHS = NO;\n\t\t\t\tCLANG_ANALYZER_NONNULL = YES;\n\t\t\t\tCLANG_ANALYZER_NUMBER_OBJECT_CONVERSION = YES_AGGRESSIVE;\n\t\t\t\tCLANG_CXX_LANGUAGE_STANDARD = \"gnu++14\";\n\t\t\t\tCLANG_CXX_LIBRARY = \"libc++\";\n\t\t\t\tCLANG_ENABLE_MODULES = YES;\n\t\t\t\tCLANG_ENABLE_OBJC_ARC = YES;\n\t\t\t\tCLANG_WARN_BLOCK_CAPTURE_AUTORELEASING = YES;\n\t\t\t\tCLANG_WARN_BOOL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_COMMA = YES;\n\t\t\t\tCLANG_WARN_CONSTANT_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_DIRECT_OBJC_ISA_USAGE = YES_ERROR;\n\t\t\t\tCLANG_WARN_DOCUMENTATION_COMMENTS = YES;\n\t\t\t\tCLANG_WARN_EMPTY_BODY = YES;\n\t\t\t\tCLANG_WARN_ENUM_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_INFINITE_RECURSION = YES;\n\t\t\t\tCLANG_WARN_INT_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_NON_LITERAL_NULL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_OBJC_LITERAL_CONVERSION = YES;\n\t\t\t\tCLANG_WARN_OBJC_ROOT_CLASS = YES_ERROR;\n\t\t\t\tCLANG_WARN_RANGE_LOOP_ANALYSIS = YES;\n\t\t\t\tCLANG_WARN_STRICT_PROTOTYPES = YES;\n\t\t\t\tCLANG_WARN_SUSPICIOUS_MOVE = YES;\n\t\t\t\tCLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE;\n\t\t\t\tCLANG_WARN_UNREACHABLE_CODE = YES;\n\t\t\t\tCLANG_WARN__DUPLICATE_METHOD_MATCH = YES;\n\t\t\t\tCODE_SIGN_IDENTITY = \"iPhone Developer\";\n\t\t\t\tCOPY_PHASE_STRIP = NO;\n\t\t\t\tDEBUG_INFORMATION_FORMAT = \"dwarf-with-dsym\";\n\t\t\t\tENABLE_NS_ASSERTIONS = NO;\n\t\t\t\tENABLE_STRICT_OBJC_MSGSEND = YES;\n\t\t\t\tGCC_C_LANGUAGE_STANDARD = gnu11;\n\t\t\t\tGCC_NO_COMMON_BLOCKS = YES;\n\t\t\t\tGCC_WARN_64_TO_32_BIT_CONVERSION = YES;\n\t\t\t\tGCC_WARN_ABOUT_RETURN_TYPE = YES_ERROR;\n\t\t\t\tGCC_WARN_UNDECLARED_SELECTOR = YES;\n\t\t\t\tGCC_WARN_UNINITIALIZED_AUTOS = YES_AGGRESSIVE;\n\t\t\t\tGCC_WARN_UNUSED_FUNCTION = YES;\n\t\t\t\tGCC_WARN_UNUSED_VARIABLE = YES;\n\t\t\t\tIPHONEOS_DEPLOYMENT_TARGET = 11.2;\n\t\t\t\tMTL_ENABLE_DEBUG_INFO = NO;\n\t\t\t\tSDKROOT = iphoneos;\n\t\t\t\tSWIFT_OPTIMIZATION_LEVEL = \"-Owholemodule\";\n\t\t\t\tVALIDATE_PRODUCT = YES;\n\t\t\t};\n\t\t\tname = Release;\n\t\t};\n\t\t14DE4D5920424E77003C7BF4 /* Debug */ = {\n\t\t\tisa = XCBuildConfiguration;\n\t\t\tbuildSettings = {\n\t\t\t\tASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;\n\t\t\t\tCODE_SIGN_STYLE = Manual;\n\t\t\t\tDEVELOPMENT_TEAM = \"\";\n\t\t\t\tINFOPLIST_FILE = A_J_Full_Screen_Image_Browser/Info.plist;\n\t\t\t\tIPHONEOS_DEPLOYMENT_TARGET = 11.0;\n\t\t\t\tLD_RUNPATH_SEARCH_PATHS = \"$(inherited) @executable_path/Frameworks\";\n\t\t\t\tPRODUCT_BUNDLE_IDENTIFIER = \"io.pigfly.A-J-Full-Screen-Image-Browser\";\n\t\t\t\tPRODUCT_NAME = \"$(TARGET_NAME)\";\n\t\t\t\tPROVISIONING_PROFILE_SPECIFIER = \"\";\n\t\t\t\tSWIFT_VERSION = 4.0;\n\t\t\t\tTARGETED_DEVICE_FAMILY = \"1,2\";\n\t\t\t};\n\t\t\tname = Debug;\n\t\t};\n\t\t14DE4D5A20424E77003C7BF4 /* Release */ = {\n\t\t\tisa = XCBuildConfiguration;\n\t\t\tbuildSettings = {\n\t\t\t\tASSETCATALOG_COMPILER_APPICON_NAME = AppIcon;\n\t\t\t\tCODE_SIGN_STYLE = Manual;\n\t\t\t\tDEVELOPMENT_TEAM = \"\";\n\t\t\t\tINFOPLIST_FILE = A_J_Full_Screen_Image_Browser/Info.plist;\n\t\t\t\tIPHONEOS_DEPLOYMENT_TARGET = 11.0;\n\t\t\t\tLD_RUNPATH_SEARCH_PATHS = \"$(inherited) @executable_path/Frameworks\";\n\t\t\t\tPRODUCT_BUNDLE_IDENTIFIER = \"io.pigfly.A-J-Full-Screen-Image-Browser\";\n\t\t\t\tPRODUCT_NAME = \"$(TARGET_NAME)\";\n\t\t\t\tPROVISIONING_PROFILE_SPECIFIER = \"\";\n\t\t\t\tSWIFT_VERSION = 4.0;\n\t\t\t\tTARGETED_DEVICE_FAMILY = \"1,2\";\n\t\t\t};\n\t\t\tname = Release;\n\t\t};\n/* End XCBuildConfiguration section */\n\n/* Begin XCConfigurationList section */\n\t\t14DE4D4120424E77003C7BF4 /* Build configuration list for PBXProject \"A_J_Full_Screen_Image_Browser\" */ = {\n\t\t\tisa = XCConfigurationList;\n\t\t\tbuildConfigurations = (\n\t\t\t\t14DE4D5620424E77003C7BF4 /* Debug */,\n\t\t\t\t14DE4D5720424E77003C7BF4 /* Release */,\n\t\t\t);\n\t\t\tdefaultConfigurationIsVisible = 0;\n\t\t\tdefaultConfigurationName = Release;\n\t\t};\n\t\t14DE4D5820424E77003C7BF4 /* Build configuration list for PBXNativeTarget \"A_J_Full_Screen_Image_Browser\" */ = {\n\t\t\tisa = XCConfigurationList;\n\t\t\tbuildConfigurations = (\n\t\t\t\t14DE4D5920424E77003C7BF4 /* Debug */,\n\t\t\t\t14DE4D5A20424E77003C7BF4 /* Release */,\n\t\t\t);\n\t\t\tdefaultConfigurationIsVisible = 0;\n\t\t\tdefaultConfigurationName = Release;\n\t\t};\n/* End XCConfigurationList section */\n\t};\n\trootObject = 14DE4D3E20424E77003C7BF4 /* Project object */;\n}\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser.xcodeproj/project.xcworkspace/contents.xcworkspacedata",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<Workspace\n   version = \"1.0\">\n   <Group\n      location = \"group:../A_J_Full_Screen_Image_Browser/animator\"\n      name = \"animator\">\n      <FileRef\n         location = \"group:FullScreenImageTransitionAnimator.swift\">\n      </FileRef>\n   </Group>\n   <Group\n      location = \"group:../A_J_Full_Screen_Image_Browser/asset\"\n      name = \"asset\">\n      <FileRef\n         location = \"group:FullScreenImageBrowser.bundle\">\n      </FileRef>\n   </Group>\n   <Group\n      location = \"group:../A_J_Full_Screen_Image_Browser/core\"\n      name = \"core\">\n      <FileRef\n         location = \"group:FullScreenImageBrowser.swift\">\n      </FileRef>\n      <FileRef\n         location = \"group:FullScreenImageBrowserViewModel.swift\">\n      </FileRef>\n      <FileRef\n         location = \"group:Image+AsyncDownload.swift\">\n      </FileRef>\n      <FileRef\n         location = \"group:MaskImageViewer.swift\">\n      </FileRef>\n      <FileRef\n         location = \"group:SingleImageViewer.swift\">\n      </FileRef>\n      <FileRef\n         location = \"group:ZoomableImageView.swift\">\n      </FileRef>\n   </Group>\n   <Group\n      location = \"group:../A_J_Full_Screen_Image_Browser/helper\"\n      name = \"helper\">\n      <FileRef\n         location = \"group:SingleImage.swift\">\n      </FileRef>\n      <FileRef\n         location = \"group:UIView+SnapShot.swift\">\n      </FileRef>\n   </Group>\n   <FileRef\n      location = \"self:\">\n   </FileRef>\n</Workspace>\n"
  },
  {
    "path": "A_J_Full_Screen_Image_Browser.xcodeproj/project.xcworkspace/xcshareddata/IDEWorkspaceChecks.plist",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<!DOCTYPE plist PUBLIC \"-//Apple//DTD PLIST 1.0//EN\" \"http://www.apple.com/DTDs/PropertyList-1.0.dtd\">\n<plist version=\"1.0\">\n<dict>\n\t<key>IDEDidComputeMac32BitWarning</key>\n\t<true/>\n</dict>\n</plist>\n"
  },
  {
    "path": "LICENSE",
    "content": "Copyright (c) 2018 Junliang Jiang Foundation (https://pigfly.github.io)\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in\nall copies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\nTHE SOFTWARE."
  },
  {
    "path": "README.md",
    "content": "<p align=\"center\">\n    <img src=\"https://github.com/pigfly/A_J_Full_Screen_Image_Browser/blob/master/assets/logo.png?raw=true\">\n</p>\n\n# A-J-Full-Screen-Image-Browser\n\n![Travis](https://img.shields.io/travis/USER/REPO.svg)\n![Code](https://img.shields.io/badge/code-%E2%98%85%E2%98%85%E2%98%85%E2%98%85%E2%98%85-brightgreen.svg)\n![Swift](https://img.shields.io/badge/Swift-%3E%3D%203.1-orange.svg)\n![npm](https://img.shields.io/npm/l/express.svg)\n\nA-J-Full-Screen-Image-Browser is an drop-in solution for full screen image and video browser\n\n## Features\n\n- [x] No Dependency, 100% iOS Native\n- [x] Support both iPad and iPhone family\n- [x] Support image resizing on different screen orientation\n- [x] Support multiple videos and images\n- [x] Image can be panned, zoomed and rotated\n- [x] Double tap to zoom all the way in and again to zoom all the way out\n- [x] Swipe to dismiss\n- [x] High level diagram\n- [x] MVVM architecture\n- [x] Full documentation\n- [x] Easy to customise\n\n## Requirements\n\n- iOS 9.0+ / macOS 10.10+ / tvOS 9.0+ / watchOS 2.0+\n- Xcode 9.0+\n- Swift 4.0+\n\n## Installation\n\n- drag and drop the entire `A_J_Full_Screen_Image_Browser` into your project\n\n## Full Usage Example\n\n```swift\nimport UIKit\n\nfinal class ViewController: UIViewController {\n\n    lazy var testVideo: MediaDownloadable = {\n        return SingleMedia(imageURL: URL(string: \"https://dummyimage.com/600&text=thumbnail\")!,\n                    isVideoThumbnail: true,\n                    videoURL: URL(string: \"http://jplayer.org/video/m4v/Big_Buck_Bunny_Trailer.m4v\")!)\n    }()\n\n    lazy var media: [MediaDownloadable] = {\n        return [testVideo,\n                SingleMedia(imageURL: URL(string: \"https://dummyimage.com/300\")!),\n                SingleMedia(imageURL: URL(string: \"https://dummyimage.com/600\")!),\n                testVideo]\n    }()\n\n    @IBAction func onButtonTapped(_ sender: UIButton) {\n        let vm = FullScreenImageBrowserViewModel(media: media)\n        let browser = FullScreenImageBrowser(viewModel: vm)\n        present(browser, animated: true, completion: nil)\n    }\n\n}\n```\n\n## AlamofireImage Support\n\n> By default, `FullScreenImageBrowser` doesn't use any 3rd library, the `SingleImage` uses `URLSession` to fetch image. However it's designed to be compatible with any networking library, one good example is [AlamofireImage](https://github.com/Alamofire/AlamofireImage)\n\nThe following code snippet shows an example how to use `AlamofireImage` to seamlessly integrated with `FullScreenImageBrowser`.\n\n```swift\nimport Foundation\nimport AlamofireImage\n\npublic class FullScreenImage: MediaDownloadable {\n    public var image: UIImage?\n    public var imageURL: URL?\n    public var videoURL: URL?\n    public var isVideoThumbnail: Bool\n\n    public init(imageURL: URL?, isVideoThumbnail: Bool = false, videoURL: URL? = nil) {\n        self.imageURL = imageURL\n        self.videoURL = videoURL\n        self.isVideoThumbnail = isVideoThumbnail\n    }\n\n    public func loadImageWithCompletionHandler(_ completion: @escaping (UIImage?, NSError?) -> Void) {\n        if let image = image {\n            completion(image, nil)\n            return\n        }\n        loadImageWithURL(imageURL, completion: completion)\n    }\n\n    // use any network calls you like\n    public func loadImageWithURL(_ url: URL?, completion: @escaping (_ image: UIImage?, _ error: NSError?) -> Void) {\n        guard let _url = url else {\n            completion(nil, NSError(domain: \"FullScreenImageBrowserDomain\",\n                                    code: -2,\n                                    userInfo: [ NSLocalizedDescriptionKey: \"Image URL not found.\"]))\n            return\n        }\n        let urlRequest = URLRequest(url: _url)\n\n        downloader.download(urlRequest) { [weak self] response in\n            debugPrint(response.result)\n\n            if let remoteImage = response.result.value {\n                self?.image = remoteImage\n                completion(remoteImage, nil)\n            } else {\n                completion(nil, NSError(domain: \"FullScreenImageBrowserDomain\",\n                                        code: -1,\n                                        userInfo: [ NSLocalizedDescriptionKey: \"Couldn't load image from remote\"]))\n            }\n        }\n    }\n}\n```\n\n## Folder Structure\n\n```shell\n├── animator\n│   └── FullScreenImageTransitionAnimator.swift\n├── asset\n│   └── FullScreenImageBrowser.bundle\n│       ├── close.png\n│       ├── close@2x.png\n│       └── close@3x.png\n├── core\n│   ├── FullScreenImageBrowser.swift\n│   ├── FullScreenImageBrowserViewModel.swift\n│   ├── MediaDownloadable.swift\n│   ├── MaskImageViewer.swift\n│   ├── SingleImageViewer.swift\n│   └── ZoomableImageView.swift\n└── helper\n    ├── SingleImage.swift\n    └── UIView+SnapShot.swift\n```\n\n| File                                 | Responsiblity                                                                        |\n|--------------------------------------|--------------------------------------------------------------------------------------|\n| animator                             | customised fade in/fade out animations with damping factors                          |\n| asset                                | customised static image asset for the full screen image/video browser navigation bar |\n| core/FullScreenImageBrowser          | manager class to be responsible for full screen image/video browser                  |\n| core/FullScreenImageBrowserViewModel | datasource and business logic for full screen image/video browser                    |\n| core/MediaDownloadable               | protocol to define images to be able to asynchronously download                      |\n| core/MaskImageViewer                 | `customised` overlay view for full screen image/video browser                        |\n| core/SingleImageViewer               | view controller to be responsible for single image rendering on the full screen      |\n| core/ZoomableImageView               | view to add support for image to zoom, pin, rotate, and animation                    |\n\n## Demo\n\n<p align=\"center\">\n    <img src=\"https://github.com/pigfly/A_J_Full_Screen_Image_Browser/blob/master/assets/demo.gif?raw=true\">\n</p>\n\n<p align=\"center\">\n    <img src=\"https://github.com/pigfly/A_J_Full_Screen_Image_Browser/blob/master/assets/demo2.gif?raw=true\">\n</p>\n\n<p align=\"center\">\n    <img src=\"https://github.com/pigfly/A_J_Full_Screen_Image_Browser/blob/master/assets/demo3.gif?raw=true\">\n</p>\n\n<p align=\"center\">\n    <img src=\"https://github.com/pigfly/A_J_Full_Screen_Image_Browser/blob/master/assets/demo4.gif?raw=true\">\n</p>\n\n<p align=\"center\">\n    <img src=\"https://github.com/pigfly/A_J_Full_Screen_Image_Browser/blob/master/assets/demo5.gif?raw=true\">\n</p>\n\n<p align=\"center\">\n    <img src=\"https://github.com/pigfly/A_J_Full_Screen_Image_Browser/blob/master/assets/demo6.gif?raw=true\">\n</p>\n\n## HLD\n\n<p align=\"center\">\n    <img src=\"https://github.com/pigfly/A_J_Full_Screen_Image_Browser/blob/master/assets/hld.png?raw=true\">\n</p>\n\n\n## Credits\n\nA-J-Full-Screen-Image-Browser is owned and maintained by the [Alex Jiang](https://pigfly.github.io). Thanks [iTMan.design](https://itman.design) for providing computational resources.\n\n## License\n\nA-J-Full-Screen-Image-Browser is released under the MIT license.\n"
  }
]