[iOS] 0. Migration Guide
This guide will help you migrate from the previous version of the CMP SDK to the current version. We'll cover the changes in method names, parameters, and new or deprecated functionality. For Demo Apps examples, please check this link.
Please be aware that this version of the CMP SDK were completely rebuilt from scratch, and so it represents a major breaking change, as all the methods were renamed, as were renamed the signatures, also now offering callbacks to almost all of the methods. In all cases, you will need to modify your code and update your dependencies to ensure that your mobile app works as expected. Additionally, is worth mentioning that all the data persisted by the previous version of our SDK on the users' devices will be erased, which will force the app to redisplay the consent layer.
Repositories
Please note that all of our repositories changed for the iOS SDK. Follow instructions below to find out where to point your dependency manager.
Also, notice that we do not provide a static version of our custom framework.
Cocoapod
On your Podfile, replace the previous line with the one below:
pod 'cm-sdk-ios-v3, '3.2.0'After changing, run this on the command-line:
pod install --repo-updateSwift Package Manager
Our XCFramework package is now hosted on https://github.com/iubenda/cm-sdk-xcframework-v3.
On XCode, go to the menu File > Add Package Dependencied and point it to the URL above.
Key Migration Points
- 
Delegate pattern: Instead of individual listeners, the new version uses a single delegate protocol (CMPManagerDelegate) for handling events. It contains 4 main events:- 
didReceiveConsent(consent: String, jsonObject: [String : Any])
 This is triggered when the consent layer was closed after the user updating his/her consents OR when invoking methods that cause changes in the consents, like acceptAll, rejectAll, acceptVendors, rejectVendors, etc. It means that the user accepted of rejected some of all of the consents, and that those were correctly saved in the device.
- 
didShowConsentLayer
 This is triggered when the consent layer was actually displayed. It means that there was not a consent valid in the device, so a new one should be collected.
- 
didCloseConsentLayer
 This is triggered when the SDK checked the need for a consent, but it was not needed and the layer was not displayed. It means that there is already a valid in the device, so a new one is not necessary and tue consent layer will not be displayed.
- 
didReceiveError
 This is triggered when the SDK operation has thrown any error.
 
- 
- Completion Handlers: Many methods now include completion handlers for asynchronous operations. Update your code to handle these callbacks appropriately.
- 
Consent String: Use exportCMPInfo()instead ofgetConsentString()to retrieve the consent information.
- Vendor and Purpose Consents: The methods to get vendor and purpose consents now return arrays of IDs. You may need to adjust your logic to handle these arrays.
- 
US Privacy String: The getUSPrivacyString()method has been deprecated. If you were using this for CCPA compliance, please note that this method is not available anymore.
- 
Consent Requirement Check: Use the new checkAndOpen(completion:)method to automatically determine if the consent layer needs to be displayed.
Methods and signatures changes
Initialization
- Old: CMPConsentTool(cmpConfig: CmpConfig).initialize()
- New: CMPManager.shared.setUrlConfig(UrlConfig)
Set UI Configuration
- Old: .withCmpViewControllerConfigurationBlock { ... }
- New: CMPManager.shared.setWebViewConfig(ConsentLayerUIConfig())
Check and Open Consent Layer
- Old: check({ ... }, isCached: Bool)
- New: checkAndOpen(completion: (Error?) -> Void)
Open Consent Layer
- Old: openView()
- New: forceOpen(completion: (Error?) -> Void)
Accept All Consents
- Old: acceptAll(onFinish: () -> Void)
- New: acceptAll(completion: (Error?) -> Void)
Reject All Consents
- Old: rejectAll(onFinish: () -> Void)
- New: rejectAll(completion: (Error?) -> Void)
Enable Purposes
- Old: enablePurposeList([String], onFinish: () -> Void)
- New: acceptPurposes([String], updatePurpose: Bool, completion: (Error?) -> Void)
Disable Purposes
- Old: disablePurposeList([String], onFinish: () -> Void)
- New: rejectPurposes([String], updateVendor: Bool, completion: (Error?) -> Void)
Enable Vendors
- Old: enableVendorList([String], onFinish: () -> Void)
- New: acceptVendors([String], completion: (Error?) -> Void)
Disable Vendors
- Old: disableVendorList([String], onFinish: () -> Void)
- New: rejectVendors([String], completion: (Error?) -> Void)
Get All Purposes
- Old: getAllPurposes() -> String
- New: getUserStatus() -> [CMPUserStatusResponse]
Get Enabled Purposes
- Old: getEnabledPurposes() -> String
- New: getUserStatus() -> [CMPUserStatusResponse]
Get All Vendors
- Old: getAllVendors() -> String
- New: getUserStatus() -> [CMPUserStatusResponse]
Get Enabled Vendors
- Old: getEnabledVendors() -> String
- New: getUserStatus() -> [CMPUserStatusResponse]
Check Purpose Consent
- Old: hasPurposeConsent(String) -> Bool
- New: getStatusForPurpose(id: String) -> [UniqueConsentStatus]
Check Vendor Consent
- Old: hasVendorConsent(String) -> Bool
- New: getStatusForVendor(id: String) -> [UniqueConsentStatus]
Export CMP String
- Old: exportCmpString() -> String
- New: exportCMPInfo() -> String
Import CMP String
- Old: importCmpString(String, completion: (Error?) -> Void)
- New: importCMPInfo(String, completion: (Error?) -> Void)
Reset Consent Data
- Old: reset()
- New: resetConsentManagementData(completion: (Error?) -> Void)
Dealing with Google Consent Mode Status
- Old: withGoogleAnalyticsCallback(analyticsListener: CmpGoogleAnalyticsInterface)
- New: getGoogleConsentModeStatus
Dealing with the onLinkClick callback
- Old: v2 used to have a whitelist feature. On the current version, the process was simplified and the user has full control over the behavior to follow according to the URL that is returned from the method.
cmpConfig.domainWhitelist = ["add your domains to be whitelisted"]
cmpManager.withOnCmpLinkClickListener({ url, decisionHandler in
//check URL and add the nav action
decisionHandler!.pointee = WKNavigationActionPolicy.allow
decisionHandler!.pointee = WKNavigationActionPolicy.cancel
// return shouldCloseWebView (true) or stay open (false)
return true
})- New: The user has full control over which behavior to follow according to the URL. Thus, the previous whitelisted URL's should be migrated to inside the callback method,
cmpManager.setLinkClickHandler { url in
    // Handle links to specific domains externally. The user has full control over which behavior to follow
    // according to the URL. The previous whitelisted URL's should be migrated to inside the callback method, 
    // and the user has the choice to use a switch-case statement, pattern matching, etc.
    if url.host?.contains("google.com") == true || 
       url.host?.contains("facebook.com") == true {
        UIApplication.shared.open(url, options: [:], completionHandler: nil)
        return true // URL handled externally
    }
    
    // Let other URLs load in the WebView
    return false
}Deprecated Methods:
- checkIfConsentIsRequired(completion: @escaping (Bool) -> Void)
- 
hasUserChoice() -> Bool
- 
hasPurposeConsent(id: String) -> Bool
- 
hasVendorConsent(id: String) -> Bool
- 
openConsentLayer(completion: @escaping (NSError?) -> Void)
- 
getAllPurposesIDs() -> [String]
- 
getEnabledPurposesIDs() -> [String]
- 
getDisabledPurposesIDs() -> [String]
- 
getAllVendorsIDs() -> [String]
- 
getEnabledVendorsIDs() -> [String]
- 
getDisabledVendorsIDs() -> [String]
- consentRequestedToday() -> Bool
- isConsentRequired() -> Bool
- getUSPrivacyString() -> String
- withCloseListener(() -> Void)
- withOpenListener(() -> Void)
- withErrorListener((CmpErrorType, String?) -> Void)
- withOnCMPNotOpenedListener(() -> Void)
- withOnCmpButtonClickedCallback((CmpButtonEvent) -> Void)
- withCmpViewControllerConfigurationBlock((UIViewController?) -> Void)
- withCmpViewConfigurationBlock((UIView) -> Void)
- withUpdateGoogleConsent(([String: String]?) -> Void
Note: The new SDK uses a shared instance (CMPManager.shared) and a delegate pattern for callbacks. Implement CMPManagerDelegate for handling events.
Migration examples
Swift
// ============================================
// Previous versions
// ============================================
override func viewDidLoad({
  super.viewDidLoad()
    let vendoradded = NSNotification.Name.CmpConsentVendorAdded;
  NotificationCenter.default.addObserver(self, selector: #selector(handleVendorAdded(notification:)), name: vendoradded, object: nil)
    NotificationCenter.default.addObserver(self, selector: #selector(handleConsentUpdated(notification:)), name: Notification.Name.CmpConsentConsentChanged, object: nil)
    NotificationCenter.default.addObserver(self, selector: #selector(handleVendorRemoved(notification:)), name: Notification.Name.CmpConsentVendorRemoved, object: nil)
    NotificationCenter.default.addObserver(self, selector: #selector(handleConsentUpdated(notification:)), name: Notification.Name.CmpConsentChanged, object: nil)
    setupCmpConfig();
private func setupCmpConfig() {
  let cmpConfig : CmpConfig = CmpConfig.shared.setup(withId: "YOUR_CODE_ID_HERE, domain: myCmpConfig.domain, appName: myCmpConfig.appName, language: myCmpConfig.language);
  cmpConfig.logLevel = CmpLogLevel.verbose;
  cmpManager = CMPConsentTool(cmpConfig: cmpConfig)
    .withErrorListener(onCMPError)
    .withCloseListener(onClose)
    .withOpenListener(onOpen)
    .withOnCMPNotOpenedListener(onCMPNotOpened)
    .withOnCmpButtonClickedCallback(onButtonClickedEvent)
    .withCmpViewControllerConfigurationBlock { viewController in
      viewController?.modalPresentationStyle = .popover
      viewController?.modalTransitionStyle = .crossDissolve
                                              }
  .initialize()
  }
// ============================================
// SDK v3 implementation
// ============================================
                                           
class DemoAppViewController: UIViewController {
    override func viewDidLoad() {
        super.viewDidLoad()
        view.backgroundColor = .white
        
        let cmpManager = CMPManager.shared
        cmpManager.setUrlConfig(UrlConfig(id: "YOUR_CODE_ID_HERE", domain: "delivery.consentmanager.net", language: "IT", appName: "CMDemoAppSwift"))
        cmpManager.setWebViewConfig(ConsentLayerUIConfig(
            position: .fullScreen,
            backgroundStyle: .dimmed(.black, 0.5),
            cornerRadius: 5,
            respectsSafeArea: true,
            allowsOrientationChanges: true
        ))
        cmpManager.setPresentingViewController(self)
        cmpManager.delegate = self
        cmpManager.checkAndOpen() { result in
            print("CMPManager initialized and open consent layer opened if necessary")
        }
    }
}
// MARK: - SDK delegates - callbacks
extension DemoAppViewController: CMPManagerDelegate {
    func didChangeATTStatus(oldStatus: Int, newStatus: Int, lastUpdated: Date?) {
        print("DemoApp received a change in the ATTStatus")
    }
    
    func didReceiveError(error: String) {
        print("DemoApp received consent layer error: \(error)")
    }
    
    func didReceiveConsent(consent: String, jsonObject: [String : Any]) {
        print("DemoApp received consent.")
    }
    
    func didShowConsentLayer() {
        print("DemoApp displayed Consent Layer.")
    }
  
    func didCloseConsentLayer()
        print("DemoApp received close consent message.")
        
        let homeView = HomeView()
        let hostingController = UIHostingController(rootView: homeView)
        self.view.window?.rootViewController = hostingController
    }
}Objective-C
// ==========================================
// Objective-C v3 implementation
// ==========================================
- (void)initializeConsentManager {
    CMPManager *cmpManager = [CMPManager shared];
    [cmpManager setDelegate:self];
    
    UrlConfig *urlConfig = [[UrlConfig alloc] initWithId:@"YOUR_CODE_ID_HERE"
                                                 domain:@"delivery.consentmanager.net"
                                               language:@"EN"
                                                appName:@"CMDemoAppObjC"];
    [cmpManager setUrlConfig:urlConfig];
    
    ConsentLayerUIConfig *uiConfig = [[ConsentLayerUIConfig alloc] initWithPosition:PositionFullScreen
                                                                    backgroundStyle:BackgroundStyleDimmed
                                                                       cornerRadius:5
                                                                   respectsSafeArea:YES
                                                          allowsOrientationChanges:YES];
    [cmpManager setWebViewConfig:uiConfig];
    
    [cmpManager setPresentingViewController:self];
    
    [cmpManager checkAndOpen:^(NSError * _Nullable error) {
        if (error) {
            NSLog(@"Error initializing CMPManager: %@", error.localizedDescription);
        } else {
            NSLog(@"CMPManager initialized and open consent layer opened if necessary");
        }
    }];
} 
                                                    






