Shortcut Items were introduced with the announcement of 3D Touch on iPhone 6S and iPhone 6S Plus. They allow users to quickly jump to a particular section or feature of an app, straight from the home screen.
Shortcut Items come in two forms: Static, and Dynamic. Static shortcut items live in the Info.plist of our project. Dynamic shortcut items are defined in code, and configured at run time. Let's take a look at how to implement static shortcut items.
The first step is to configure the shortcut items in our Info.plist file:
Then, we need to implement one new function on our app delegate. We're using JLRoutes here (covered in Bite #62) to run some code when a given URL is passed in.
Then we'll call the provided completionHandler, with a Bool indicating whether we were able to handle the item.
Note: At publish time of this Bite, Xcode did not offer a way to test shortcut items in the Simulator. To test shortcut items' functionality, this project by Conrad Kramer was used.
UPDATE: We covered Dynamic Shortcut Items in Bite #88. Head there to read about how to modify an app'sshortcut items in code, at runtime.
NSURLSession is the heart of networking in Foundation. Let's take a look at a few ways to use it.
The basic idea is we'll create a new NSURLSession, and then use it to create NSURLSessionTasks that will make the actual HTTP requests. We'll start by creating a new session:
We'll can use a data task like above, then use this function to grab the downloaded data. This will be called multiple times, so we'll need to keep an NSMutableData reference around to append to.
Imagine an app that downloads and plays movies. If a user begins downloading a movie, it'd be great if the file could continue downloading even if the user presses the home button and sends the app to the background. NSURLSessionDownloadTask can help. Let's take a look.
First we create a new background configuration and session. Then we use that session to create a new download task:
We can grab the downloaded data pretty easily, by implementing this function from NSURLSessionDownloadDelegate:
funcURLSession(session:NSURLSession,downloadTask:NSURLSessionDownloadTask,didFinishDownloadingToURLlocation:NSURL){print("Movie data written to \(location)")}
Last but not least, we'll need to implement a function from NSURLSessionTaskDelegate. In it we'll call finishTasksAndInvalidate on the session. This ensures everything is cleaned up and the session lets go of us (self) as its delegate.
It's very common for an app to need to regularly check for new data. Background Fetch allows us to do just that, while running in the background.
The system will periodically wake our app up and ask it to fetch new data. Once awoken, we'll be able to run whatever code we need to check for or fetch new data.
Lastly, we'll need to tell the system what the result of our fetch was. The system will use our response to decide how frequently our app needs to fetch.
Let's start by going to the capabilities tab of our project, and turning on Background Modes. Then we'll check the box next to the Background fetch mode.
Next, the code. We'll set a minimum background fetch interval in our didFinishLaunchingfunction.
The performFetchWithCompletionHandlerfunction is the last piece of the puzzle. We fetch any new data that might exist, and call the provided completionHandler with an enum describing the result.
Calling the completion handler will also trigger the system to take a new snapshot of our app for the multitasking interface, so we should make sure we've updated any views that need it beforehand.
To test everything, we can use the Debug > Simulate Background Fetch command in the Simulator, or create a background fetch-specific scheme like we covered in Bite #65.
That's just the beginning though. Locksmith supports almost every feature Keychain has to offer. It also embraces protocols, and protocol extensions. Let's look at an example of how to improve our own types, making them easier to store in the Keychain.
Let's say we have a simple struct in our app to represent user accounts. We'll implement a couple of protocols from Locksmith on it:
Note how we not only give the cell a detail button, but we pass in a closure to be run when the detail button is tapped. We can even use a custom UIView as an accessory like we did with the switch in our first example.
We used a simple string as our footer here, but we can easily use a view instead:
Section(rows:[],footer:.View(footerView)).
Static does a great job separating concerns. Rows don't know anything about the actual UITableViewCells that eventually get shown to represent them. Static uses the .Value1 cell type by default and comes with a few other standard cell types. We can easily add our own by implementing the CellType protocol on any of our own UITableViewCell subclasses.
UIImage is a deceptively powerful part of UIKit. Let's take a look at some of the different ways we can use it:
Template Images
Let UIKit do the heavy lifting by using template images. UIKit only looks at the alpha channel, and draws the image using the tint color of the view it's contained in.
UIImages can be animated. A UIImage can contain many images within it, as well as a duration. Put the containing image in a UIImageView and call startAnimating to see it.
Pattern Images
This one's fun. We can create a UIColor from a UIImage. The resulting "color" will be our image, tiled. We can then use this UIColor as a background color of a view.
UIColor(patternImage:UIImage("bg-pattern")!)
Stretchable Images
Sometimes we don't want to write a bunch of Core Graphics code to implement things like rounded corners, inner shadows, etc. UIImage has a great feature for implementing resizable (also sometimes called 9-patch) images.
We want to implement a fancy button. All we need is a tiny little 9x9 image. In code, we'll tell UIKit to load the image, and create a resizable image from it.
This will cause our navigation bar to slide on and off screen as the user scrolls, just like in Safari for iOS.
Empty Back Button
We want a back button with no text. We could set the title of our first view controller to an empty string, but then our first view controller would be title-less. We can get the best of both worlds by giving the first view controller a custom back button item with an empty title. We'll need to do this before we push on a new view controller.
UIActivityViewController is one of the main ways we can allow our apps to talk with each other, as well as allow our users to share content with the world. Let's take a look. UIActivityViewController accepts an array of activityItems which can be of many types:
The activity view controller will intelligently use each item. For example: On Twitter, the text will become the text of a tweet. The URL and image will be appended and shared natively as an image attached to the tweet.
UIActivityViewController is pretty clever. For example, it's smart enough to know if the NSURL we hand it is to a video in the user's Photos library. In such a case, it responds by letting the user share the video with services like Vimeo. We can also pass in an NSData containing the actual video.
We can even allow printing our content by wrapping it in a UISimpleTextPrintFormatter and passing that in:
Finally, we can use the completeWithItemsHandlerclosure to learn if the user actually performed a share or action in our activity view controller, as well as what specific activity they chose.
activityVC.completionWithItemsHandler={(activityType,completed,items,error)inguardcompletedelse{print("User cancelled.");return}print("Completed With Activity Type: \(activityType)")ifactivityType==UIActivityTypePostToFacebook{print("Shared on Facebook")}}
Instead of the standard red pin graphic 📍, we'd like to show a fun little treasure chest image at the spot of some secret treasure!
To do this, take advantage of MKAnnotationView's.image property. We can throw a regular UIImage at it to get exactly the behavior we're looking for. All we need to do is implement one delegate function and return an MKAnnotationView from it. MKAnnotationViews get reused just like UITableViewCells and UICollectionViewCells.