How to import Objective-C framework in Swift framework

In a normal swift project we would create a bridging-header and import our objective-C framework in the bridging-header.

In the case of a framework you will need to create a custom swift module.

Let’s say you have an objective-C framework called MyFramework.

Firstly, add your framework to the project:

Then you need to add an aggregate to the project and target list.

Enter the name of the objective-c framework as the product name.

Then go to Build Phases > + > New Run Script Phase.

Enter the following script:

Make sure you set path of the header file to the path the header file of your objective-C framework. You can find the here: https://github.com/AnumQ/public_scripts/blob/master/module.sh

BUILD the aggregate MobileFramework by selecting it from the list of schemes.

Now we must include the framework in our project. We do this by adding the MobileFramework module to the Target Dependencies of our main project (framework).

Now build your MyCustomFramework project.

Simply use the the framework in your swift file by adding the import statement:

Vola!

Resign an IPA file including frameworks

Run these commands from the terminal in the folder where you have YOUR_IPA.ipa file.

  1. unzip -q YOUR_IPA.ipa
  2. rm -rf Payload/YOUR_IPA.app/_CodeSignature
  3. rm -rf Payload/YOUR_IPA.app/Frameworks/X.framework/_CodeSignature
  4. codesign -f -s “iPhone Distribution: XXX (XXXX)” Payload/YOUR_IPA.app/Frameworks/X.framework
  5. security cms -D -i PRODUCTION.mobileprovision > pr.plist
  6. /usr/libexec/PlistBuddy -x -c ‘Print :Entitlements’ pr.plist > YOUR_Plist.plist
  7. codesign -f -s “iPhone Distribution: XXX (XXXX)” –entitlements YOUR_Plist.plist Payload/YOUR_IPA.app
  8. zip -qr ResignedApp.ipa Payload SwiftSupport Symbols

Safe area fix guide for earlier versions of iOS

Safe area layout is introduced in iOS11 but what can we do if we are supporting < iOS 11 and we have a xib view that is initiated in a ViewController and somehow the upper part of the xib view appears behind the NavigationBar??? To fix this, simply add this line in your viewDidLoad.

self.edgesForExtendedLayout = UIRectEdge.init(rawValue: 0)

For more information visit EdgesForExtendedLayout at Apple.

Embed a UIViewController in a UINavigationController Programmatically

We know how to embed a ViewController in a NavigationController in Storyboard. However, I came across a scenario where I had to initiate it programmatically. The code:

let viewController = MyViewController()
let nav = UINavigationController(rootViewController: viewController)
self.navigationController?.present(nav, animated: true, completion: nil)

Unique ID on iOS Devices

Problem!

We all know now that identifierForVendor doesn’t remain unique on iOS Devices. It changes when the user deletes all of the apps from the same vendor.

According to Apple:

The value in this property remains the same while the app (or another app from the same vendor) is installed on the iOS device. The value changes when the user deletes all of that vendor’s apps from the device and subsequently reinstalls one or more of them. The value can also change when installing test builds using Xcode or when installing an app on a device using ad-hoc distribution. Therefore, if your app stores the value of this property anywhere, you should gracefully handle situations where the identifier changes.

Please see Apple Documentation for more information.

Solution

To get the unique Id we need to generate it first through CFUUID and save it on the device using KeyChain, in this case our SwiftKeychainWrapper will do the job. This way, even when the user deletes the app the UUID can be fetched using the UNIQUE_KEY that was originally used to save the UUID.

import UIKit
import SwiftKeychainWrapper

class ViewController: UIViewController {

    @IBOutlet weak var uuidValue: UILabel!
    private let UNIQUE_KEY = "mySuperDuperUniqueId"
    override func viewDidLoad() {
        super.viewDidLoad()
        let uniqueDeviceId: String? = KeychainWrapper.standard.string(forKey: UNIQUE_KEY)
        
        guard uniqueDeviceId != nil else {
            let uuid = generateUuid()
            let saveSuccessful: Bool = KeychainWrapper.standard.set(uuid, forKey: UNIQUE_KEY)
            if saveSuccessful {
                uuidValue.text = uuid
            } else {
                fatalError("Unable to save uuid")
            }
            return
        }
        uuidValue.text = uniqueDeviceId!
    }
    
    private func generateUuid() -> String {
        let uuidRef: CFUUID = CFUUIDCreate(nil)
        let uuidStringRef: CFString = CFUUIDCreateString(nil, uuidRef)
        return uuidStringRef as String
    }
}