data:image/s3,"s3://crabby-images/0abea/0abea1049937deacc46c6a8f9154baf556c65457" alt="Multiple Xcode Targets with CocoaPods"
During one of our recent projects, we were tasked with creating an app that can change its implementation depending upon the type of the app that the client wants to export. There are many reasons why someone may want to follow this setup; like using the same codebase for minor differences in the apps or any iOS developer to be able to keep his development and production environments separate etc. Our client had the same requirement but he needed the application in 4 different variants which made the situation all the more interesting. This is usually done by creating multiple Xcode targets in the Xcode. A target is an end product of the “build” command in the Xcode. In this tutorial, we are going to create iOS Multiple Xcode targets with Cocoapods albeit with 2 types using Xcode 10 and Swift 4.2.
These two types of apps will use a single view application containing a UITableView. We’re going to load different ULRs based on the project type and load the appropriate content. The API that we are going to use is a project called: JSONPlaceholder by taking the Photos and Users end points. Let’s start with creating a new project:data:image/s3,"s3://crabby-images/137bd/137bd5036e66748ec3f50a7468aad6da3be790ca" alt="single-app-view"
data:image/s3,"s3://crabby-images/7378a/7378ad95dd247e6be2449dd1b49fadaa7cd2856a" alt="single-view-app"
data:image/s3,"s3://crabby-images/89ea0/89ea013a90be7e7c9cd8c3dd7d06f364c0e012c6" alt="single-app-view step 3"
data:image/s3,"s3://crabby-images/c23bd/c23bd41d8bd2ffbcc928ae0df0109facc880ea35" alt="multiple target samples"
struct Photo {
let photoTitle: String!
let photoUrl: String!
}
Now, creating the method to fetch the contents:fileprivatefunc loadScreenContent() {
let urlString = "https://jsonplaceholder.typicode.com/photos"
iflet url = URL(string: urlString) {
let request = URLRequest(url: url)
let session = URLSession.shared
let task = session.dataTask(with: request) { (data, urlResponse, error) in
if (error != nil) {
//we got error from service
//handle error here
} else {
//No error
iflet jsonString = String(data: data!, encoding:String.Encoding.utf8) {
self.photos = self.getPhotosFromJsonString(jsonString: jsonString)
DispatchQueue.main.async {
self.tableView.reloadData()
}
}
}
}
task.resume()
}
}
And the parser helper method:fileprivatefunc getPhotosFromJsonString(jsonString: String) -> [Photo] {
var photosToReturn = [Photo]()
do {
let photosArray = tryJSONSerialization.jsonObject(with: jsonString.data(using: String.Encoding(rawValue: String.Encoding.utf8.rawValue))!, options: .allowFragments) as! [[String:Any]]
for photoDictionay in photosArray {
var title = ""
iflet currentPhotoTitle = photoDictionay["title"] as? String {
title = currentPhotoTitle
}
var thumbnailUrl = ""
iflet currentPhotoImageUrl = photoDictionay["thumbnailUrl"] as? String {
thumbnailUrl = currentPhotoImageUrl
}
let photo = Photo(photoTitle: title, photoUrl: thumbnailUrl)
photosToReturn.append(photo)
}
} catch {
//an exception has occured
}
return photosToReturn
}
The next step for us is to show these photos in the UITableViewCell. In order to show the image, we are going to integrate an image cache library named Kingfisher to help us with the process. We are going to do it using a CocoaPods. Again in this tutorial, we are not going to cover what is CocoaPods, if you don’t know about it, you can go to this website and check it out but for this tutorial’s sake, it’s being assumed that you know about it already.
Let’s start by introducing CocoaPods to this project. In order to do so, open the terminal and go to the project folder and run this command:$ pod init
This will create a pod file for you to update. Let’s update the pod file:
$ open.
To open the folder containing the pod file. After opening the pod file, past the following line:
pod 'Kingfisher', '~> 4.0'data:image/s3,"s3://crabby-images/42ed2/42ed2a6637a2655d294f9ad91e13613f08c9b909" alt="storyboard"
Going back to terminal, install the pod:
$ pod install
Now that the pod has been installed, let’s go back to Finder and open the MultipleTargetsSample.xcworkspace to reload the project. Now, the project should look something like this:data:image/s3,"s3://crabby-images/7230e/7230eea2a7de09a1c7a0ac8dde983d7b76b01137" alt="adding UITableview"
func configureCell(photo: Photo) {
imageNameLabel.text = photo.photoTitle
iflet url = URL(string: photo.photoUrl) {
customimageView.kf.setImage(with: url)
}
}
Now everything has been set up, the last thing that we need to do is to load the UITableView. Putting in the UITableViewDataSource:func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
returnphotos.count
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
iflet cell = tableView.dequeueReusableCell(withIdentifier: "PhotosTableViewCell", for: indexPath) as? PhotosTableViewCell {
cell.selectionStyle = .none
cell.configureCell(photo: photos[indexPath.row])
}
returnUITableViewCell()
}
Now, everything is set up, we are ready to run the app for our first type to see to see how it loads:data:image/s3,"s3://crabby-images/56f91/56f912e636a7542cde0367fcca3c9670dd1605b7" alt="kingfisher podfile"
- Create a new target
- Update the pod file to support multiple Xcode targets
- Create flags for each target
- Updating the View Controller to run code according to the selected target
Creating a new target:
The first step is to create a new target. In order to do so, we need to duplicate the target as it is. In order to do so, go to your project’s settings, select the target, right click on it and select the duplicate option.data:image/s3,"s3://crabby-images/f79f3/f79f3a19135b94ef76394759668e5e383ac585ea" alt="pod install screen"
data:image/s3,"s3://crabby-images/a0025/a0025c13c8dcd02f1646303a7b8e5dc22cf0af54" alt="photos look"
This process creates schemes for each target as well.
data:image/s3,"s3://crabby-images/ec238/ec2387c21dd5f352ca9f6642cdff60283a6b779c" alt="creating a new target"
data:image/s3,"s3://crabby-images/282e0/282e0cc3a4d6c8197ae8ba4740f9da2ebe265556" alt="after duplication"
data:image/s3,"s3://crabby-images/69c48/69c487ef5654c9576d8959739cf4faf288b98778" alt="create scheme for each target"
Update the pod file to support multiple Xcode targets
Now that we have set up iOS multiple targets, our next step is to update the pods file so that it supports both the targets. Our existing pod file looks something like thistarget 'MultipleTargetsSample' do
# Comment the next line if you're not using Swift and don't want to use dynamic frameworks
use_frameworks!
# Pods for MultipleTargetsSample
pod 'Kingfisher', '~> 4.0'
end
Since pod file is actually a Ruby file, what we need to do is to create a single object and share it between the both targets. It’ll look something like this:def sharedpods
pod 'Kingfisher', '~> 4.0'
end
target 'MultipleTargetsPhotos' do
sharedpods
end
target 'MultipleTargetsUsers' do
sharedpods
end
Now, go to the Terminal back and run the pod install command again and you’ll be good to go for both of these targets.Create flags for each target
Now that the pods are working for each target, the next step is for us to start the process of differentiating the code for each target. In the olden days, with the Objective-C, we used to do it with the Preprocessor Macros but with the Swift that part is no longer valid. What arrives as a substitute, is a compile-time attributes. In order to add this, let’s go to the Photos target -> Build Settings -> Swift Compiler – Custom Flags and add a custom flag called “-DPhotosType”data:image/s3,"s3://crabby-images/de7ba/de7ba55975badbad51c841955ad0c9cfedff43d1" alt="Rename the target"
Updating the View Controller
By this point we have duplicated the targets, updated the pods to share it among the multiple Xcode targets and add custom flags to the targets. Now, the last thing that we need to do is update the view controllers to update the view controller to adjust the code based on the target. In order to do so we’ll be use the #if statements. The part will have multiple small steps:- Updating the title
- Writing the code to fetch the users from the server
- Updating the UITableViewDataSource
#if PhotosType
self.title = "Photos"
#else
self.title = "Users"
#endifAll, well and done, updating the code to fetch the users. Let's start by updating the loadScreenContent method and replace it with this code:
fileprivate func loadScreenContent() {
var urlString: String!
#if PhotosType
urlString = "https://jsonplaceholder.typicode.com/photos"
#else
urlString = "https://jsonplaceholder.typicode.com/users"
#endif
if let url = URL(string: urlString) {
let request = URLRequest(url: url)
let session = URLSession.shared
let task = session.dataTask(with: request) { (data, urlResponse, error) in
if (error != nil) {
//we got error from service
//handle error here
} else {
//No error
if let jsonString = String(data: data!, encoding:String.Encoding.utf8) {
#if PhotosType
self.photos = self.getPhotosFromJsonString(jsonString: jsonString)
#else
self.users = self.getUsersFromJsonString(jsonString: jsonString)
#endif
DispatchQueue.main.async {
self.tableView.reloadData()
}
}
}
}
task.resume()
}
}Now creating a model for the Usersstruct Users {
let userName: String!
}
Parsing the results from the JSON string:
fileprivate func getUsersFromJsonString(jsonString: String) -> [Users] {
var usersToReturn = [Users]()
do {
let usersArray = try JSONSerialization.jsonObject(with: jsonString.data(using: String.Encoding(rawValue: String.Encoding.utf8.rawValue))!, options: .allowFragments) as! [[String:Any]]
for userDictionay in usersArray {
var name = ""
if let currentUserName = userDictionay["name"] as? String {
name = currentUserName
}
let user = Users(userName: name)
usersToReturn.append(user)
}
} catch {
//an exception has occured
}
return usersToReturn
}
The only thing that remains is now updating the UITableViewDataSource to accomodate the changes for both targets now. Replace the UITableViewDataSource to this code now:
Now, let’s change the target to users to see if it works or no.func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
#if PhotosType
return photos.count
#else
return users.count
#endif
}
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
#if PhotosType
if let cell = tableView.dequeueReusableCell(withIdentifier: "PhotosTableViewCell", for: indexPath) as? PhotosTableViewCell {
cell.selectionStyle = .none
cell.configureCell(photo: photos[indexPath.row])
return cell
}
#else
if let cell = tableView.dequeueReusableCell(withIdentifier: "UserTableViewCell", for: indexPath) as? UserTableViewCell {
cell.selectionStyle = .none
cell.configureCell(user: users[indexPath.row])
return cell
}
#endif
return UITableViewCell()
}
data:image/s3,"s3://crabby-images/e04e2/e04e215e8f1a4fbb4586ce49b548472f9fade833" alt="rename the target screen"
Works flawlessly!
Changing back to the photos to see if it worksdata:image/s3,"s3://crabby-images/58577/585772b1d0e879e9f1e0302400f18fb2096106af" alt="flag creation for each target"