Students will build a simple alarm app to practice intermediate table view features, protocols, the delegate pattern, Codable, and UserNotifications.
Students who complete this project independently are able to:
- Implement a master-detail interface
- Implement the
UITableViewDataSource
protocol - Implement a static
UITableView
- Create a custom
UITableViewCell
- Write a custom delegate protocol
- Wire up view controllers to model object controllers
- Work with the
Date
andCalendar
of Swift - Add staged data to a model object controller
- Create model objects that conform to the
Codable
protocol - Create model object controllers that use
JSONEncoder
andJSONDecoder
for data persistence - Schedule and cancel
UserNotifications
- Create custom protocols
- Implement protocol functions using protocol extensions to define protcol function behavior across all conforming types
Set up a basic List-Detail view hierarchy using a UITableViewController for a AlarmListTableViewController and a AlarmDetailTableViewController. Use the provided screenshots as a reference.
- Add a
UITableViewController
scene that will be used to list alarms - Embed the scene in a
UINavigationController
- Add an 'Add' system bar button item to the navigation bar
- Add a class file
AlarmListTableViewController.swift
and assign the scene in the Storyboard - Add a
UITableViewController
scene that will be used to add and view alarms
- note: We will use a static table view for our Alarm Detail view, static table views should be used sparingly, but they can be useful for a table view that will never change, such as a basic form. You can make a table view static by selecting the table view, going to the Attribute Inspector, and changing the Content dropdown from Dynamic Prototypes to Static Cells.
- Add a show segue from the Add button from the first scene to the second scene.
- Add a show segue from the prototype cell form the first scene to the second scene.
- Add a class file
AlarmDetailTableViewController.swift
and assign the scene in the Storyboard
Build a custom table view cell to display alarms. The cell should display the alarm time, the alarm name, and have a switch that will toggle whether or not the alarm is enabled.
It is best practice to make table view cells reusable between apps. As a result, you will build a SwitchTableViewCell
rather than an AlarmTableViewCell
that can be reused any time you want a cell with a switch.
- Add a new
SwitchTableViewCell.swift
as a subclass of UITableViewCell. - Configure the prototype cell in the Alarm List Scene in
Main.storyboard
to be an instance ofSwitchTableViewCell
- Design the prototype cell as shown in the screenshots: two labels, one above the other, with a switch to the right.
- note: Stack views are great. Think about using a horizontal stack view that has a vertical stack view and a switch inside of it. The vertical stack view will have two labels in it.
- Create an IBOutlet to the custom cell file for the label named
timeLabel
. - Create an IBOutlet to the custom cell file for the label named
nameLabel
. - Create an IBOutlet to the custom cell file for the switch named
alarmSwitch
. - Create an IBAction for the switch named
switchValueChanged
which you will implement using a custom protocol later in these instructions.
Build a static table view as the detail view for creating and editing alarms.
- Static table views do not need to have UITableViewDataSource functions implemented. Instead, you can create outlets and actions from your prototype cells directly onto the view controller (in this case
AlarmDetailTableViewController
) as you would with other types of views. - If you haven't already, go to your Storyboard, select your detail table view and in the Attributes Inspector change the Style to Grouped and the Sections to 3. By default each section will have 3 cells in it. You can delete 2 of the cells in each section.
- In section 1, drag a date picker onto the prototype cell and add proper constraints.
- In section 2, drag a text field onto the prototype cell and add the proper constraints and placeholder text.
- In section 3, drag a button onto the prototype cell and add the proper constraints and title. This button will be used to enable/disable existing alarms.
- Create IBOutlets for the three items listed above and create an IBAction for the button titled
enableButtonTapped
. - If you haven't already, add a bar button item to the right side of the navigation bar, change the System Item to Save in the Attributes Inspector, and create an IBAction called
saveButtonTapped
.
- You will need to add a Navigation Item to the Navigation Bar before you can add the bar button item.
Create an Alarm.swift
which will contain our Alarm model object. Alarms should be able to represent a name, whether they are enabled (on or off), a unique identifier, a time when the alarm will go off, and a string representation of that time.
- Add a property called
fireDate
which stores aDate
representing the time the alarm will go off , a property calledname
of typeString
, and a property calledenabled
of typeBool
that we will set to true if the alarm is enabled and false otherwise. - Add a property called uuid of type
String
. Give the uuid property a default value ofUUID().uuidString
in the memberwise initializer. This will make it so that when called, the initializer won't ask you to provide a value for the uuid property. A UUID is a Universally Unique Identifier. Theuuid
on the Alarm object will be used later to schedule and cancel local notifications - Add a computed property called
fireTimeAsString
which will return aString
representation of the time you want the alarm to fire. This is simply for the UI. *note: Use Apple's DateFormater class to return a String from your existingfireDate
property. *Please Read: https://developer.apple.com/documentation/foundation/dateformatter
Create an AlarmController
model object controller that will manage and serve Alarm
objects to the rest of the application.
- Create an
AlarmController.swift
file and define a newAlarmController
class. - Add an
alarms
array property with an empty array as a default value. - Create an
addAlarm(fireDate: Date, name: String, enabled: Bool)
function that creates an alarm, adds it to thealarms
array, and returns the alarm. - Create an
update(alarm: Alarm, fireDate: Date, name: String, enabled: Bool)
function that updates an existing alarm's fire time and name. - Create a
delete(alarm: Alarm)
function that removes the alarm from thealarms
array
- note: There is no 'removeObject' function on arrays. You will need to find the index of the object and then remove the object at that index. Refer to documentation if you need to know how to find the index of an object.
- note: You will need to conform your
Alarm
model to theEquatable
protocol and implement the proper equality check.
- Create a static
shared
property that stores a shared instance.
- note: Review the syntax for creating shared instance properties
Add mock alarm data to the AlarmController. Using mock data can be very useful. Once there is mock data development teams can serialize work -- i.e. some can work on the views with visible data while others work on implementing the controller logic. This is a quick way to get objects visible so you can begin building the views.
There are many ways to add mock data to model object controllers. We will do so using a computed property.
- Create a
mockAlarms:[Alarm]
computed property that holds a number of stagedAlarm
objects
- Initialize a small number of
Alarm
objects to return with varying properties
- When you want mock data, set self.alarms to self.mockAlarms in the
AlarmController
initializer. Remove it when you no longer want mock data.
Wire up the Alarm List Table View and implement the property observer pattern on the SwitchTableViewCell
class.
Fill in the table view data source functions required to display the view.
*note: these implementation will be very similar to your Journal application with the exception of the cellForRowAt
. Insead of setting the cell.textLabel.text
property directly from the tableViewController, you will pass an alarm object into a variable within your custom cell class i.e. cell.alarm = alarm
Your custom cell should follow the 'updateViews' pattern for updating the view elements with the details of a model object. To follow this pattern, the developer adds an 'updateViews' function that checks for a model object. The function updates the view with details from the model object.
- Add a property
var alarm: Alarm?
to yourSwitchTableViewCell
class. *note: This will act very similar to the way we have used variables as "Landing Pads" or "Mail boxes" in Detail ViewControllers - Add an
updateViews()
function that updates the labels to the time and name of the alarm, and updates thealarmSwitch.isOn
property so that the switch reflects the proper alarmenabled
state. - Add a
didSet
observer on the alarm property, and call theupdateViews()
function you just made in it. - On your
AlarmListTableViewController
fill in the two requiredUITableViewDataSource
functions, using thealarms
array fromAlarmController.shared
. In thetableView(UITableView, cellForRowAt: IndexPath)
data source function you will need to cast your dequeued cell as aSwitchTableViewCell
and set the cell'salarm
property. Make sure you use the right alarm from thealarms
array inAlarmController
. - Implement the
UITableViewDataSource
tableView(_:, commit:, forRowAt:)
method to enable swipe-to-delete. Be sure to call the appropriateAlarmController
method before deleting the row.
- At this point you should be able to run your project and see your table view populated with your mock alarms. You should be able to delete rows and segue to a detail view (this detail view won't actually display an alarm yet since we haven't implemented the
prepare(for segue: UIStoryboardSegue, sender: Any?)
function, but the segue should still occur). Also note that you can toggle the switch, but that theenabled
property on the model object the cell is displaying isn't actually changing.
Write a protocol for the SwitchTableViewCell
to delegate handling a toggle of the switch to the AlarmListTableViewController
, adopt the protocol, and use the delegate function to mark the alarm as enabled or disabled and reload the cell.
- Create a custom protocol named
SwitchTableViewCellDelegate
to the top of theSwitchTableViewCell
class file - Define a
switchCellSwitchValueChanged(cell: SwitchTableViewCell)
function - Add a weak, optional delegate property on the SwitchTableViewCell
- note:
weak var delegate: SwitchTableViewCellDelegate?
- note: If the compiler throws an error, it is likely because your protocol must be restricted to class types.
- Update the
switchValueChanged(_:)
IBAction to check if a delegate is assigned, and if so, call the delegate protocol function - Adopt and conform to the protocol in the
AlarmListTableViewController
class. - In the
cellForRowAt
function in theAlarmListTableViewController
set self (the AlarmListTableViewController in this instance) as the delegate of each cell. - Go to your
AlarmController
class and add atoggleEnabled(for alarm: Alarm)
function that will switch theenabled
property of thealarm
in your function parameter to true if it is false, and false if it is true. - Go back to your
AlarmListTableViewController
class and implement theswitchCellSwitchValueChanged(cell:)
delegate function to capture the alarm, toggle the alarm's enabled property using the function you just made inAlarmController
, and reload the table view.
Create functions on the detail table view controller to display an existing alarm and setup the view properly.
- Add an
alarm
property of typeAlarm?
toAlarmDetailTableViewController
. This will hold an alarm if the view is displaying an existing alarm and will be nil if the view is being used to create a new alarm. - Add a variable named
alarmIsOn
of type Bool with an initial value oftrue
to the top of yourAlarmDetailTableViewController
class. *note: Make sure to set thealarmIsOn
property equal to thealarm.enabled
property in the case that the alarm object exists. - Create a private
updateViews()
function that will populate the date picker and alarm title text field with the current alarm's date and title. This function will set the enable button to say "On" if the alarm inself.alarm
is enabled and "Off" if it is disabled. You may consider changing background color and font color properties as well to make the difference between the two button states clear. *note: As the initialalarm
property will be nil (when we are creating a new alarm), you will need to use the isAlarmOn property you created earlier to set up the button's UI initially. *note: You must guard against the alarm being nil, or the view controller's view not yet being loaded and properly handle these cases. - Call
updateViews()
inviewDidLoad()
.
Fill in the prepare(for segue: UIStoryboardSegue, sender: Any?)
function on the AlarmListTableViewController
to properly prepare the next view controller for the segue.
- In the
AlarmListTableViewController
, add an if statement in theprepare(for segue: UIStoryboardSegue, sender: Any?)
function that checks if the segue's identifier matches the identifier of the segue that goes from a cell to the detail view. - Get the destination view controller from the segue and cast it as an
AlarmDetailTableViewController
. - Get the indexPath of the selected cell from the table view.
- Use
indexPath.row
to get the correct alarm that was tapped from theAlarmController.shared.alarms
array. - Set the
alarm
property on the destination view controller equal to the alarm from the previous step.
- If the compiler presents an error when trying to do this, you either forgot to cast the destination view controller as an
AlarmDetailTableViewController
or forgot to give theAlarmDetailTableViewController
a property titlealarm
of typeAlarm?
. - At this point you should be able to run your project and see your table view populated with your mock alarms, displaying the proper switch state. You should also be able to delete rows, and segue to a detail view from a cell. This detail view should display the proper time of the alarm, the proper title, and the proper state of the enable/disable button.
Fill in the saveButtonTapped
function on the detail view so that you can add new alarms and edit existing alarms.
- Unwrap
alarm
and if there is an alarm, call the.update(alarm: , name: , fireDate: , enabled: )
function from theAlarmController
and pass in the tite from the text field, the fireDate from the datePicker, and the enabled from thealarmIsOn
property. - If there is no alarm, call the
addAlarm(fireDate: , name: , enabled: )
function to create and add a new alarm.
- note: You should be able to run the project and have what appears to be a fully functional app. You should be able to add, edit, delete, and enable/disable alarms. We have not yet covered how to alert the user when time is up, or persist the alarms.
Make sure your Alarm
object conforom to the Codable protocol so that we can persist alarms across app launches using JSONEncoder and JSONDecoder.
- Adopt the Codable protocol on your Alarm Model. You should review Codable Protocol in the documentation before continuing.
Add persistence using apple's FileManager API, Codable, and JSONEncoder and Decoder to the AlarmController
. This will require three function fileUrl() -> URL
, saveToPersistentStore()
, and loadFromPersistentStore() -> [Alarm]
. This will follow the pattern use for persistence in Journal with a data type of Alarm instead of Entry.
- Add a private, static, computed property called
fileUrl() -> URL
which returns the correct path to the alarms file in the app's documents directory as described above. - Write a private function called
saveToPersistentStorage()
that will save the current alarms array to a file using FileManager. - Write a function called
loadFromPersistentStorage()
that will load saved Alarm objects and set self.alarms (source of truth) to the results - Call the
loadFromPersistentStorage()
function when the AlarmController is initialized or in the AppDelegatesapplication(_:didFinishLaunchingWithOptions:)
- Call the
saveToPersistentStorage()
any time that the list of alarms is modified *note: the list of alarms is modified in all of the CRUD fucntions for this app.
- note: You should now be able to see that your alarms are saved between app launches.
Register for local notifications when the app launches.
- In the
AppDelegate.swift
file, importUserNotifications
. Then in theapplication(_:didFinishLaunchingWithOptions:)
function, request notification authorization on an instance ofUNUserNotificationCenter
.
- note: See UserNotifications Documentation for furthur instrution: https://developer.apple.com/documentation/usernotifications/asking_permission_to_use_notifications
You will need to schedule local notifications each time you enable an alarm and cancel local notifications each time you disable an alarm. Seeing as you can enable/disable an alarm from both the list and detail view, we normally would need to write a scheduleUserNotifications(for alarm: Alarm)
function and a cancelUserNotifications(for alarm: Alarm)
function on both of our view controllers. However, using a custom protocol and a protocol extension, we can write those functions only once and use them in each of our view controllers as if we had written them in each view controller.
You will need to heavily reference Apples documentation on UserNotifications: https://developer.apple.com/documentation/usernotifications
- In your
AlarmController
file, but outside of the class, create aprotocol AlarmScheduler
. This protocol will need two functions:scheduleUserNotifications(for alarm:)
andcancelUserNotifications(for alarm: Alarm)
. - Below your protocol, create a protocol extension,
extension AlarmScheduler
. In there, you can create default implementations for the two protocol functions. - Your
scheduleUserNotifications(for alarm: Alarm)
function should create an instance ofUNMutableNotificationContent
and then give that instance a title and body. You can also give that instance a default sound to use when the notification goes off usingUNNotificationSound.default()
. - After you create your
UNMutableNotificationContent
, create an instance ofUNCalendarNotificationTrigger
. In order to do this you will need to createDateComponents
using thefireDate
of youralarm
. *note: Use thecurrent
property of theCalendar
class to call a method which returns dateComponents from a date.
- note: Be sure to set
repeats
in theUNCalendarNotificationTrigger
initializer totrue
so that the alarm will repeat daily at the specified time.
- Now that you have
UNMutableNotificationContent
and aUNCalendarNotificationTrigger
, you can initialize aUNNotificationRequest
and add the request to the notification center object of your app.
- note: In order to initialize a
UNNotificationRequest
you will need a unique identifier. If you want to schedule multiple requests (which we do with this app) then you need a different identifier for each request. Thus, use theuuid
property on yourAlarm
object as the identifier.
- Your
cancelLocalnotification(for alarm: Alarm)
function simply needs to remove pending notification requests using theuuid
property on theAlarm
object you pass into the function.
- note: Look at documentation for
UNUserNotificationCenter
and see if there are any functions that will help you do this. https://developer.apple.com/documentation/usernotifications/unusernotificationcenter/1649517-removependingnotificationrequest
- Conform your AlarmController Class to the
AlarmScheduler
protocol. Notice how the compiler does not make you implement the schedule and cancel functions from the protocol? This is because by adding an extension to the protocol, we have created default implementation of these functions for all classes that conform to the protocol. - In your
toggleEnabled(for alarm: Alarm)
function, you will need to schedule a notification if the switch is being turned on, and cancel the notification if the switch is being turned off. You will also need to cancel the notification when you delete an alarm.
The last thing you need to do is set up your app to notify the user when an alarm goes off and they still have the app open. In order to do this we are going to use the UNUserNotificationCenterDelegate
protocol.
- Go to your
AppDelegate.swift
file and have yourAppDelegate
class adopt theUNUserNotificationCenterDelegate
protocol. - Then in your
application(_:didFinishLaunchingWithOptions:)
function, set the delegate of the notification center to equalself
.
- note:
UNUserNotificationCenter.current().delegate = self
- Then call the delegate method
userNotificationCenter(_:willPresent:withCompletionHandler:)
and use thecompletionHandler
to set yourUNNotificationPresentationOptions
.
- note:
completionHandler([.alert, .sound])
The app should now be finished. Run it, look for bugs, and fix anything that seems off.
Please refer to CONTRIBUTING.md.
© DevMountain LLC, 2015-2016. Unauthorized use and/or duplication of this material without express and written permission from DevMountain, LLC is strictly prohibited. Excerpts and links may be used, provided that full and clear credit is given to DevMountain with appropriate and specific direction to the original content.