Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Method swizzling in swift 4 [duplicate]

Swizzling in Swift 4 no longer works.

Method 'initialize()' defines Objective-C class method 'initialize', which is not permitted by Swift

This is something I have found a solution to so wanted to leave the questions and answer for others.

like image 356
Christopher Rex Avatar asked Sep 22 '17 09:09

Christopher Rex


1 Answers

initialize() is no longer exposed: Method 'initialize()' defines Objective-C class method 'initialize', which is not permitted by Swift

So the way to do it now is to run your swizzle code via a public static method.

e.g

In the extension: (This extension is used in the kickstarted open source code: https://github.com/kickstarter/ios-oss/blob/master/Library/DataSource/UIView-Extensions.swift)

private var hasSwizzled = false

extension UIView {
    final public class func doBadSwizzleStuff() {
        guard !hasSwizzled else { return }

        hasSwizzled = true
        swizzle(self) /* This is pseudo - run your method here */
    }
}

In the app delegate: (This method is used in the kickstarted open source code: https://github.com/kickstarter/ios-oss/blob/7c827770813e25cc7f79a28fa151cd713efe936f/Kickstarter-iOS/AppDelegate.swift#L33)

func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: UIApplicationLaunchOptionsKey: Any]?) -> Bool 
{
    UIView.doBadSwizzleStuff()
}

Another way is to use a singleton:

extension UIView {
    static let shared : UIViewController = {
        $0.initialize()
        return $0
    }(UIViewController())

    func initialize() {
        // make sure this isn't a subclass
        guard self === UIViewController.self else { return }

        let swizzleClosure: () = {
            UIViewController().swizzle() /* This is pseudo - run your method here */
        }()
        swizzleClosure
    }
}

func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: UIApplicationLaunchOptionsKey: Any]?) -> Bool 
{
    _  = UIViewController.shared
}
like image 193
Christopher Rex Avatar answered Oct 29 '22 10:10

Christopher Rex