So far, you learned how to implement these features in the SwiftCollectionView application:
Today you will learn how to implement these feature in the SwiftCollectionView application:
- Display the number of selected cells in the navigation bar.
- Share selected collection view cell’s photos with friends.
Take a look at this video. It shows how the app will function on a real iOS device; from the moment you tap the share button to the moment you tap the Mail Composer’s Send button.
Video 8-1: App running on an iPhone 4s device
The image below shows what you’ll see when you run the app on the iOS Simulator and tap the share button.
Figure 8-1: Output on the iPhone 4s Simulator
Before You Begin
Before you begin, I assume the following:
- You created a new branch from the collection-view-plist branch and you called it share-photos.
- You will use Xcode Version 6.1.1 to implement the above features in the SwiftCollection project.
- You’re a member of the Apple iOS Developer Program. If you aren’t, then you won’t be able to test the mailing feature of this version of the SwiftCollectionView app.
- In Xcode Version 6.1.1 you have to update the lines that display errors in the MasterViewController.swift file by adding an! at the end of self.collectionView. Here’s how to fix the first one.
Change the statement on line 32 from this:
self.collectionView.backgroundColor = UIColor(patternImage: UIImage(named: "purty_wood")!)
to this:
self.collectionView!.backgroundColor = UIColor(patternImage: UIImage(named: "purty_wood")!)
For remaining lines that have an error, add the ! as shown above.
Create and Connect The Share Button in IB
Here are the steps to create the share button on the navigation bar.
- Click the Main.storyboard file to load it in Interface Builder.
- Drag a Bar Button from the Object Library and drop it on the left corner of the Photos scene’s navigation Bar.
- In the Attributes inspector, select Action from the Identifier menu.
- Use the Assistant editor to create and connect an IBOutlet variable for the button.
- Use the Assistant editor to create and connect an IBAction function for the shareButton; call it shareButtonTapped method. You should create the function at the end of the MasterViewController.swift file.
Add Two rightBarButtons on The Nav Bar
You have to modify the Photos scene’s navigation Bar so it displays two buttons on the right side of the navigation bar-see image below. Interface Builder allows you to add only one button on either end of the navigation bar. They are called leftBarButton and rightBarButton. To add two rightBarButtons on the navigation bar, you’ll have to add code in the MasterViewController.swift file’s viewDidLoad() function.
Start by deleting the Edit button from the Photos scene’s navigation bar. Also, delete text from the navigation’s Title field. Switch to the MasterViewController.swift file and delete the Edit button’s IBOutlet variable.
Here is the code to create and add the rightBarButtons on the photos scene’s navigation bar. Add the code in the MasterViewController.swift file’s viewDidLoad() function.
// Set two properties of the counterLabel object counterLabel.textColor = UIColor(red:1.00, green:0.00, blue:0.50 ,alpha:1.00) // pink color counterLabel.text = "0 photos selected" //1. Create two bar Bar Button Items, connect only one of them to an IBAction function photoCounterButton = UIBarButtonItem(customView: counterLabel) editButton = UIBarButtonItem(title: "Edit", style: .Plain, target:self, action:"editButtonTapped:") //2. Add the buttons in the buttonArray variable let buttonArray = [editButton, photoCounterButton] //3. Put the buttons on the Photos scene's nav bar self.navigationItem.rightBarButtonItems = buttonArray
Remember; the photoCounterButton’s sole purpose is to display the number of collection view cells selected, so that’s why we did not connect it to an IBAction function. Later, you’ll add code in the MasterViewController.swift file to update the button’s counterLabel properties.
The last thing you have to do to complete the task of setting up the rightBarButtons on the Photos scene’s navigation bar is, add these variable declaration statements at the top of the MasterViewController.swift file.
var photoCounterButton: UIBarButtonItem! var editButton: UIBarButtonItem! var shareEnabled = false var selectedCellItems = [String]() var counterLabel = UILabel()
Run the application say, in the iPhone 4s simulator. You will only see the Share and edit button on the navigation bar. Later, when you add code to update the photoCounter button, the photoCounter button’s text shows up on the Photos view’s navigation bar-see above image. The Edit button should work as before; however, the share button doesn’t yet.
How The Share Button Will Work
Before I show you the share button’s code; I’ll explain how it will work. In a nutshell, the share button’s job is to put the app in “edit mode” so you can select one more collection view cells’ photos and email them to one or more friends. The share button is connected to the shareButtonTapped() function. The code you’ll add in the function will perform several tasks. Here is a list of the most important ones:
- Put the app in sharing mode.
- Create an object called mailComposer from the MFMailComposeViewController class.
- Construct an email for the MailComposer view.
- Display the mailComposer’s view on the device’s screen.
- Wait for you to enter one or more email addresses in the To: field and to tap/click the send button.
- Once you tap/click the mailComposer view’s Send button. The app responds by checking the device’s type. If its a real device; the app sends your email to the recipient’s email address. You will hear a swooshing sound-assuming your device’s speaker isn’t muted. If the device is an iOS Simulator, the app will display a message in an alert view.
A note about the last item in the above list. If you try to send an email using an iOS Simulator, the app will crash. That’s because the iOS Simulator can’t send emails; only a real iOS device can.
Configure The MasterViewController Class to Send Mail
When you tap the share button, code in the shareButtonTapped() function will pass control to an instance of the MFMailComposerViewController class. In code, we’ll call that instance mailComposer. Its job is to present its view on the device’s screen so you can send mail. Before you can use the MFMailComposerViewController class in the MasterViewController.swift file, you have to configure it to use the MFMailComposerViewController class.
Start by executing these steps, which import the MessageUI.framework in the project. The framework has three header files, one of them is the MFMailComposerViewController.h file.
- Click the root project folder in the project navigator panel.
- Click the Build Phases tab.
- Expand the Link Binary With Libraries item. Click the + button.
- Type mes in the popup window’s search box.
- Select the MessageUI.framework in the search results list, then click the Add button. Xcode add the framework in the project.
Now, you have to conform the MasterViewController.swift class file to the MFMailComposeViewControllerDelegate protocol. You do that by changing the class file’s header to this:
class MasterViewController: UICollectionViewController, MFMailComposeViewControllerDelegate {
Finally, you have to import the MessageUI.h file in the MasterViewController.swift file. You do that by adding this import statement in the MasterViewController.swift file.
import MessageUI
Ok, so you’ve configured the MasterViewController.swift class file to send mail. You are ready to add code in the shareButtonTapped() function.
Add Code in The shareButtonTapped() Function
Here is the first set of code to add in the shareButtonTapped() function. It checks to see if the device is an iOS Simulator or not. If it is, then code in the if statement block is executed. If it isn’t, then only the print() statement is executed.
@IBAction func shareButtonTapped(sender: AnyObject) { if deviceIsSimulator() { let alertView = UIAlertController(title: nil, message: "Your device can't send mail", preferredStyle: .Alert) alertView.addAction(UIAlertAction(title: "Ok", style: .Default, handler: nil)) presentViewController(alertView, animated: true, completion: nil) // This prevent code below from executing return } print("The device is not a simulator and shareEnabled = \(shareEnabled)") }
Notice how the deviceIsSimulator() function is called in the if statement. Here is the code to implement the function. Add the code below the shareButtonTapped() function’s code.
// This function check to see if the current device is an iPhone or iPad Simulator. // If so, then return the Boolean value true, otherwise return false func deviceIsSimulator() -> Bool { var isSimulator = false if UIDevice.currentDevice().model == "iPhone Simulator" || UIDevice.currentDevice().model == "iPad Simulator" { isSimulator = true } return isSimulator }
Test The Share Button Code
Now is a good time to run the app in the iPhone 4s simulator to test the share button code. You should see output shown in Figure 8-1 above on the simulator’s screen when you click the share button. Test the app on a real device, you’ll have to enroll in Apple’s iOS Developer Program. Assuming you’re a member of Apple’s iOS Developer Program, here’s how to test the app on your device.
Connect your real device to your Mac, hit the run button in Xcode. Wait for Xcode to launch the app on your device. Once the app is running on your device, tap the share button. You won’t see the above alert view message on your device’s screen; however, you will see the print() function’s message in Xcode’s Debugger window. The reason for this is because the if statement evaluates to false and its code block isn’t executed.
Add More Code in The shareButtonTapped() Function
Now, delete the print() function statement you added in the shareButtonTapped() function. Next, add this code in the function. Add it below the if statement block’s closing } bracket.
if shareEnabled == true { } else { }
Ok, above code snippet is saying this, if the app is in sharing mode, do this. Otherwise, do that. When is the app put in sharing mode and taken out of sharing mode? The first time you tap the share button, the app is put in sharing mode. When you tap the share button again; the app is taken out of sharing mode. Here is the code to add in the above if statement block.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 |
[swift gutter="false"]// Create a Mail Composer View Controller object let mailComposer = MFMailComposeViewController() // Set the mailComposer delegate property mailComposer.mailComposeDelegate = self // Set the mailComposer subject property mailComposer.setSubject("Check out these photos") // Create a variable and assign it the HTML table header tag var emailBody = " <table style='border-collapse: collapse; border: 0px solid; cellpadding='3'> <tbody> <tr>" // This for loop set up rows for the HTML table for var i = 0; i < selectedCellItems.count; i++ { var imageUrl = selectedCellItems[i] emailBody += " <td><img src='\(imageUrl)' width='75' height='75' /></td> " if i == 6 { emailBody += "</tr> <tr>" } if i == 13 { emailBody += "</tr> <tr>" } if i == 20 { emailBody += "</tr> <tr>" } if i == 27 { emailBody += "</tr> <tr>" } if i == 34 { emailBody += "</tr> <tr>" } if i == 41 { emailBody += "</tr> <tr>" } if i == 54 { emailBody += "</tr> " } } // Append the HTML table footer tags to the emailB emailBody += "</tbody> </table> " // Set the mailComposer message boady and isHTML parameters mailComposer.setMessageBody(emailBody, isHTML: true) // Take the app out of sharing mode shareEnabled = false // Set the collection view to single cell selection mode self.collectionView!.allowsMultipleSelection = false // Empty out the selectedCellItems array selectedCellItems.removeAll(keepCapacity: false) // Update the counter label on the nav bar updateCounterLabel() // Show the mail composer view controller's view on the device's screen self.presentViewController(mailComposer, animated: true, completion: nil) // Refresh the collection view, so previously highlighted cells gets unhighlighted self.collectionView!.reloadData() [/swift] Here is the code to add in the else block. [swift gutter="false"]// Put the app in sharing mode shareEnabled = true // Empty out the selectedCellItems array selectedCellItems.removeAll(keepCapacity: false) // Allow the user to select multiple cells in the collection view self.collectionView!.allowsMultipleSelection = true // Refresh the collection view, so previously highlighted cells gets unhighlighted self.collectionView!.reloadData() [/swift] Code you entered in the shareButtonTapped() function calls the updateCounterLabel(). Here's the code to implement it. [swift gutter="false"]// This function is called in the shareButtonTaped() function and the // collection view's didSelectItemAtIndexPath() function func updateCounterLabel() { // Update the counterLabel text property counterLabel.text = "\(selectedCellItems.count) photos selected" // Update the counterLabel's width so all text is shown counterLabel.sizeToFit() }[/swift] Now, here's the code to implement the delegate function of the MFMailComposeViewControllerDelegate protocol. Add it in the MasterViewController.swift file. [swift gutter="false"]// This function is fired when the user tap the Send or Cancel button on the mail composer view's navigation bar func mailComposeController(controller: MFMailComposeViewController!, didFinishWithResult result: MFMailComposeResult, error: NSError!) { // Close the mail composer view controller's view self.dismissViewControllerAnimated(true, completion: nil) }[/swift] That's it. You are done adding code in the shareButtonTapped() function. Take a look at the table below; its a recap of what code you entered in the shareButtonTapped() does. <table style="border-collapse: collapse; border: 1px solid;"> <tbody> <tr> <td valign="top">Check to see if the device the app is running on is an iOS Simulator. If it is, then do the following: <ul> <li>Create and display an alert view on the simulator's screen.</li> <li>Don't execute remaining code in the shareButtonTapped() function.</li> </ul> Check the boolean variable, shareEnabled; if it is equal to false, then do the following: <ul> <li>Put the app in sharing mode by setting the shareEnabled variable to true.</li> <li>Set the collection view to multiple selection mode by setting its allowsMultipleSelection property to true.</li> <li class="p1"><span class="s1">Refresh the collection view, so previously highlighted cells are unhighlighted.</span></li> </ul> Check the boolean variable, shareEnabled; if it is equal to true, then do the follow: <ul> <li>Use the MFMailComposeViewController class to create a object called mailComposer.</li> <li>Set the mailComposer's subject property.</li> <li>Create an object called emailBody and initialize it with HTML text.</li> <li>In a for loop, append more HTML text to the emailBody variable.</li> <li>Outside the for loop, append more HTML text to the emailBody variable.</li> <li>Set the mailComposer's body and isHTML properties via the mailComposer's setMessageBody() function.</li> <li>Take the app out of sharing mode by setting the shareEnabled variable to false.</li> <li>Set the collection view to single selection mode by setting its allowsMultipleSelection property to false.</li> <li>Empty out the selectedCellItems array variable.</li> <li>Update properties of the navigation bar's photoCounterButton by calling the updateCounterLabel() function.</li> <li>Show the mail composer view controller's view on the device's screen.</li> <li>Refresh the collection view, so previously highlighted cells gets unhighlighted.</li> </ul> </td> </tr> </tbody> </table> <span style="font-size: 14pt;"><strong>Replace The didSelectItemAtIndexPath() Function's Code</strong></span> You have to replace the collection view didSelectItemAtIndexPath() function's code with this: [swift gutter="false"] override func collectionView(collectionView: UICollectionView, didSelectItemAtIndexPath indexPath: NSIndexPath) { // Put the selected collection view cell's photo url in a variable selectedPhotoUrl = cvDataSource[indexPath.row] if shareEnabled == true { // Add an object in the selectedCellItems array selectedCellItems.insert(selectedPhotoUrl, atIndex: 0) // Update the counter label on the nav bar updateCounterLabel() } else { // Pass control to the PhotoViewController self.performSegueWithIdentifier("photoDetail", sender:self) } }[/swift] <span style="font-size: 14pt;"><strong>Replace The didDeselectItemAtIndexPath() Function's Code</strong></span> The last task you have to do is replace the collection view didDeselectItemAtIndexPath() function's code with this: [swift gutter="false"]override func collectionView(collectionView: UICollectionView, didDeselectItemAtIndexPath indexPath: NSIndexPath) { // Put the deselected cell's photo url in a variable var itemToDelete = cvDataSource[indexPath.item] // Typecast the selectedCellItems array to an Objective-C array var objectArray: NSArray = selectedCellItems as NSArray // Typecast the objectArray to an Objective-C NSMutableArray var mutableObjects = NSMutableArray(array: objectArray) if shareEnabled == true { // Delete an element from the NSMutableArray mutableObjects.removeObject(itemToDelete) // Empty out the selectedCellItems array selectedCellItems.removeAll(keepCapacity: false) // Use double-casting to convert the Objective-C NSMutableArray to a Swift Array, // before assigning it to the selectedCellItems array selectedCellItems = mutableObjects as NSArray as [String] // Update the counter label on the nav bar updateCounterLabel() /************** DEBUG CODE **************** println("\nDeleted item: \(itemToDelete)\n") println(finalMutableArray) ********************************************/ } }[/swift] Use the Source Control menu to commit changes made to the project. <h4>Test The Share Button's Code, Again</h4> Run the app say, in the iPhone 4s simulator. You should see output shown in Figure 8-1 above, when you tap the navigation bar's share button. If you have a real device, connect it to your Mac, and test the share button's code. You should see output shown in the QuickTime video presented at the beginning of this workshop. Here's what I did in the video <ul> <li>I tapped the share button once.</li> <li>I tapped several collection view cells.</li> <li>I tapped the share button again. This pop open the Mail Composer View Controller's view over the Photos view.</li> <li>I entered an email address in the To: field.</li> <li>I tapped the Send button. The Mail Composer View Controller sent the email to the email address I entered in the To: field and close its view.</li> </ul> <h4>What's Next?</h4> That's it, you've entered necessary code in the MasterViewController.swift file to display the number of selected cells in the navigation bar, and to share selected collection view cell's photo with friends. Next, week you will learn how to do the following: <ul> <li>Display the collection view cells in multiple sections.</li> <li>Display section headers and footers in the collection view.</li> <li>Update the Delete button's code to enable the app user to delete one or more cells from each section of the collection view.</li> </ul> Until then, happy coding! :) |
No responses yet