swiftでiPhone標準の時計アプリを作ろう 完成コード アラーム

2019年9月12日木曜日 更新:2019年09月17日

swift アラーム 時計

t f B! P L
Xcode 10.3
ios 12.4
swit5

完成品

UI

今回はViewが多いですがなるべく標準に似せて設置しました。

コード

AppDelegate

  1. import UIKit
  2. import UserNotifications
  3. @UIApplicationMain
  4. class AppDelegate: UIResponder, UIApplicationDelegate {
  5. var window: UIWindow?
  6. func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
  7. // Override point for customization after application launch.
  8. // 通知許可の取得
  9. UNUserNotificationCenter.current().requestAuthorization(
  10. options: [.alert, .sound, .badge]){
  11. (granted, _) in
  12. if granted{
  13. UNUserNotificationCenter.current().delegate = self
  14. }
  15. }
  16. return true
  17. }
  18. func applicationWillEnterForeground(_ application: UIApplication) {
  19. // Called as part of the transition from the background to the active state; here you can undo many of the changes made on entering the background.
  20. let center = UNUserNotificationCenter.current()
  21. center.getDeliveredNotifications { (notifications: [UNNotification]) in
  22. for notification in notifications {
  23. _ = AlarmVC.shared.getAlarm(from: notification.request.identifier)
  24. NotificationCenter.default.post(name: Notification.Name("NotificationIdentifier"), object: nil)
  25. }
  26. }
  27. }
  28. }
  29.  
  30. extension AppDelegate: UNUserNotificationCenterDelegate{
  31. func userNotificationCenter(_ center: UNUserNotificationCenter, willPresent notification: UNNotification, withCompletionHandler completionHandler: @escaping (UNNotificationPresentationOptions) -> Void) {
  32. // アプリ起動中でもアラートと音で通知
  33. completionHandler([.alert, .sound])
  34. let uuid = notification.request.identifier
  35. _ = AlarmVC.shared.getAlarm(from: uuid)
  36. NotificationCenter.default.post(name: Notification.Name("NotificationIdentifier"), object: nil)
  37. }
  38. func userNotificationCenter(_ center: UNUserNotificationCenter, didReceive response: UNNotificationResponse, withCompletionHandler completionHandler: @escaping () -> Void) {
  39. let identifier = response.actionIdentifier
  40. if identifier == "snooze"{
  41. let snoozeAction = UNNotificationAction(
  42. identifier: "snooze",
  43. title: "Snooze 5 Minutes",
  44. options: []
  45. )
  46. let noAction = UNNotificationAction(
  47. identifier: "stop",
  48. title: "stop",
  49. options: []
  50. )
  51. let alarmCategory = UNNotificationCategory(
  52. identifier: "alarmCategory",
  53. actions: [snoozeAction, noAction],
  54. intentIdentifiers: [],
  55. options: [])
  56. UNUserNotificationCenter.current().setNotificationCategories([alarmCategory])
  57. let content = UNMutableNotificationContent()
  58. content.title = "Snooze"
  59. content.sound = UNNotificationSound.default
  60. content.categoryIdentifier = "alarmCategory"
  61. let trigger = UNTimeIntervalNotificationTrigger(timeInterval: 5 , repeats: false)
  62. let request = UNNotificationRequest(identifier: "Snooze", content: content, trigger: trigger)
  63. UNUserNotificationCenter.current().add(request) { (error) in
  64. if let error = error {
  65. print(error.localizedDescription)
  66. }
  67. }
  68. }
  69. let uuid = response.notification.request.identifier
  70. _ = AlarmVC.shared.getAlarm(from: uuid)
  71. NotificationCenter.default.post(name: Notification.Name("NotificationIdentifier"), object: nil)
  72. completionHandler()
  73. }
  74. }

AlarmVC

  1. import UIKit
  2. import UserNotifications
  3. class AlarmVC: UIViewController,UITableViewDelegate,UITableViewDataSource {
  4. static let shared = AlarmVC()
  5. var appDelegate = UIApplication.shared
  6.  
  7. @IBOutlet weak var tableView: UITableView!
  8. var userDefaults = UserDefaults.standard
  9. var index:Int!
  10.  
  11. var timeArray:[AlarmTimeArray] = []
  12. override func viewDidLoad() {
  13. super.viewDidLoad()
  14. tableView.allowsSelectionDuringEditing = true
  15. tableView.allowsSelection = false
  16. tableView.register(UINib(nibName: "AlarmTimeCell", bundle: nil), forCellReuseIdentifier: "AlarmTimeCell")
  17. self.navigationItem.setLeftBarButton(self.editButtonItem, animated: true)
  18. timeLoad()
  19. tableView.reloadData()
  20. NotificationCenter.default.addObserver(self, selector: #selector(self.methodOfReceivedNotification(notification:)), name: Notification.Name("NotificationIdentifier"), object: nil)
  21. }
  22. @objc func methodOfReceivedNotification(notification: Notification) {
  23. timeLoad()
  24. DispatchQueue.main.async {
  25. self.tableView.reloadData()
  26. }
  27. }
  28.  
  29. func timeLoad(){
  30. if let timeArrayData = UserDefaults.standard.object(forKey: "timeArray") as? Data {
  31. if let getTimeArray = try! NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(timeArrayData) as? [AlarmTimeArray] {
  32. timeArray = getTimeArray
  33. }
  34. }
  35. }
  36. override func setEditing(_ editing: Bool, animated: Bool) {
  37. super.setEditing(editing, animated: animated)
  38. tableView.setEditing(editing, animated: animated)
  39.  
  40. }
  41. func setCellLabel(index:Int) -> String{
  42. if timeArray[index].repeatLabel == "Never" {
  43. return timeArray[index].label
  44.  
  45. }else{
  46. return timeArray[index].label+","+timeArray[index].repeatLabel
  47. }
  48. }
  49. func getAlarm(from uuid: String){
  50. timeLoad()
  51. guard let alarm = timeArray.first(where: { $0.uuidString == uuid }) else {return }
  52. if alarm.week.isEmpty {
  53. alarm.onOff = false
  54. }
  55. saveDate()
  56. let center = UNUserNotificationCenter.current()
  57. center.removeAllDeliveredNotifications()
  58. }
  59. func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  60. return timeArray.count
  61. }
  62. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  63. let cell = tableView.dequeueReusableCell(withIdentifier: "AlarmTimeCell") as! AlarmTimeCell
  64. cell.timeLabel.text = getTime(date: timeArray[indexPath.row].date)
  65. cell.label.text = setCellLabel(index: indexPath.row)
  66. cell.sw.isOn = timeArray[indexPath.row].onOff
  67. cell.editingAccessoryType = .disclosureIndicator
  68.  
  69. return cell
  70. }
  71. func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
  72. // 画面遷移の準備
  73. if tableView.isEditing {
  74. index = indexPath.row
  75. performSegue(withIdentifier: "showAlarmAdd", sender: nil)
  76. }
  77. }
  78. func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCell.EditingStyle, forRowAt indexPath: IndexPath) {
  79. if editingStyle == .delete {
  80. UNUserNotificationCenter.current().removePendingNotificationRequests(withIdentifiers: [timeArray[indexPath.row].uuidString])
  81. timeArray.remove(at: indexPath.row)
  82. tableView.deleteRows(at: [indexPath], with: .fade)
  83. saveDate()
  84. }
  85. }
  86. func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
  87. return 72
  88. }
  89. @IBAction func addButton(_ sender: Any) {
  90. self.performSegue(withIdentifier: "showAlarmAdd", sender: nil)
  91. }
  92. func getTime(date:Date) -> String {
  93. let f = DateFormatter()
  94. f.timeStyle = .short
  95. f.locale = Locale(identifier: "ja_JP")
  96. return f.string(from: date)
  97. }
  98. func saveDate(){
  99. let timeArrayData = try! NSKeyedArchiver.archivedData(withRootObject: timeArray, requiringSecureCoding: false)
  100. userDefaults.set(timeArrayData, forKey: "timeArray")
  101. userDefaults.synchronize()
  102. }
  103. override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
  104. if segue.identifier == "showAlarmAdd"{
  105. guard let nvc = segue.destination as? UINavigationController else {return}
  106. guard let vc = nvc.topViewController as? AlarmAddVC else {return}
  107. vc.delegate = self
  108. vc.isEdit = tableView.isEditing
  109. if tableView.isEditing {
  110. vc.alarmTime = timeArray[index]
  111. }
  112. }
  113. }
  114. }
  115.  
  116.  
  117. extension AlarmVC:AlarmAddDelegate{
  118. func AlarmAddVC(alarmAdd: AlarmAddVC, alarmTime: AlarmTimeArray) {
  119. if tableView.isEditing {
  120. timeArray[index] = alarmTime
  121. }else{
  122. timeArray.append(alarmTime)
  123. }
  124. timeArray.sort(){$0.date < $1.date}
  125. saveDate()
  126. self.setEditing(false, animated: false)
  127. tableView.reloadData()
  128. }
  129. func AlarmAddVC(alarmDelete: AlarmAddVC, alarmTime: AlarmTimeArray) {
  130. self.setEditing(false, animated: false)
  131. timeArray.remove(at: index)
  132. saveDate()
  133. UNUserNotificationCenter.current().removePendingNotificationRequests(withIdentifiers: [timeArray[index].uuidString])
  134.  
  135. }
  136. func AlarmAddVC(alarmCancel:AlarmAddVC){
  137. self.setEditing(false, animated: false)
  138. }
  139. }

AlarmTimeCell

  1. import UIKit
  2.  
  3. protocol AlarmTimeCellDelegate {
  4. func alarmTimeCell(switchTappe:UITableViewCell,isOn:Bool)
  5. }
  6. class AlarmTimeCell: UITableViewCell {
  7.  
  8. @IBOutlet weak var timeLabel: UILabel!
  9. @IBOutlet weak var label: UILabel!
  10. @IBOutlet weak var sw: UISwitch!
  11. override func awakeFromNib() {
  12. super.awakeFromNib()
  13. accessoryView = sw
  14. }
  15. }
  16.  

AlarmAddVC

  1. import UIKit
  2. import UserNotifications
  3. protocol AlarmAddDelegate {
  4. func AlarmAddVC(alarmAdd:AlarmAddVC,alarmTime:AlarmTimeArray)
  5. func AlarmAddVC(alarmDelete:AlarmAddVC,alarmTime:AlarmTimeArray)
  6. func AlarmAddVC(alarmCancel:AlarmAddVC)
  7. }
  8.  
  9. class AlarmAddVC: UIViewController ,UITableViewDelegate,UITableViewDataSource{
  10. @IBOutlet weak var datePicker: UIDatePicker!
  11. @IBOutlet weak var tableView: UITableView!
  12. var delegate:AlarmAddDelegate!
  13. var alarmTime:AlarmTimeArray = AlarmTimeArray()
  14. var isEdit: Bool = false
  15. var titleText = ["Repeat","Label","Sound"]
  16. override func viewDidLoad() {
  17. super.viewDidLoad()
  18. datePicker.date = alarmTime.date
  19. registerCell(cellName: "AlarmSnoozeCell")
  20. registerCell(cellName: "AlarmAddCell")
  21. registerCell(cellName: "AlarmDeleteCell")
  22. tableView.tableFooterView = UIView()
  23. }
  24. override func viewWillAppear(_ animated: Bool) {
  25. super.viewWillAppear(animated)
  26. if let indexPathForSelectedRow = tableView.indexPathForSelectedRow {
  27. tableView.deselectRow(at: indexPathForSelectedRow, animated: true)
  28. }
  29. }
  30. //cell登録
  31. func registerCell(cellName:String){
  32. tableView.register(UINib(nibName: cellName, bundle: nil), forCellReuseIdentifier: cellName)
  33. }
  34. func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  35. switch section {
  36. case 0:
  37. return 4
  38. case 1:
  39. return 1
  40. default:
  41. return 0
  42. }
  43. }
  44. func numberOfSections(in tableView: UITableView) -> Int {
  45. return isEdit ? 2:1
  46. }
  47. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  48. switch indexPath.section {
  49. case 0:
  50. switch indexPath.row{
  51. case 0:
  52. let cell = tableView.dequeueReusableCell(withIdentifier: "AlarmAddCell") as! AlarmAddCell
  53. cell.titleLabel.text = titleText[indexPath.row]
  54. cell.subTitleLabel.text = alarmTime.repeatLabel
  55. return cell
  56. case 1:
  57. let cell = tableView.dequeueReusableCell(withIdentifier: "AlarmAddCell") as! AlarmAddCell
  58. cell.titleLabel.text = titleText[indexPath.row]
  59. cell.subTitleLabel.text = alarmTime.label
  60. return cell
  61. case 2:
  62. let cell = tableView.dequeueReusableCell(withIdentifier: "AlarmAddCell") as! AlarmAddCell
  63. cell.titleLabel.text = titleText[indexPath.row]
  64. cell.subTitleLabel.text = "Default"
  65. cell.selectionStyle = .none
  66.  
  67. return cell
  68. case 3:
  69. let cell = tableView.dequeueReusableCell(withIdentifier: "AlarmSnoozeCell") as! AlarmSnoozeCell
  70. cell.delegate = self
  71. cell.snoozeSwitch.isOn = alarmTime.snooze
  72. return cell
  73. default:
  74. break
  75. }
  76. case 1:
  77. let cell = tableView.dequeueReusableCell(withIdentifier: "AlarmDeleteCell") as! AlarmDeleteCell
  78. cell.delegate = self
  79. return cell
  80. default:
  81. break
  82. }
  83. return UITableViewCell()
  84. }
  85. func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
  86. switch indexPath.section {
  87. case 0:
  88. switch indexPath.row {
  89. case 0:
  90. self.performSegue(withIdentifier: "showRepeat", sender: nil)
  91. case 1:
  92. performSegue(withIdentifier: "showLabel",sender: nil)
  93. break
  94. case 2:break
  95. default:break
  96. }
  97. default:
  98. break
  99. }
  100. }
  101. public func tableView(_ tableView: UITableView, heightForHeaderInSection section: Int) -> CGFloat {
  102. if section == 1 {
  103. return 50
  104. }else{
  105. return 0
  106. }
  107. }
  108. //アラーム設定時間を保存
  109. @IBAction func saveButton(_ sender: Any) {
  110. alarmSet()
  111. delegate.AlarmAddVC(alarmAdd: self, alarmTime: alarmTime)
  112. dismiss(animated: true, completion: nil)
  113. }
  114. //スヌーズ設定
  115. func setCategories(){
  116. let snoozeAction = UNNotificationAction(
  117. identifier: "snooze",
  118. title: "Snooze 5 Minutes",
  119. options: []
  120. )
  121. let noAction = UNNotificationAction(
  122. identifier: "stop",
  123. title: "stop",
  124. options: []
  125. )
  126. var alarmCategory:UNNotificationCategory!
  127. if alarmTime.snooze {
  128. alarmCategory = UNNotificationCategory(
  129. identifier: "alarmCategory",
  130. actions: [snoozeAction, noAction],
  131. intentIdentifiers: [],
  132. options: [])
  133. }else{
  134. alarmCategory = UNNotificationCategory(
  135. identifier: "alarmCategory",
  136. actions: [],
  137. intentIdentifiers: [],
  138. options: [])
  139. }
  140. UNUserNotificationCenter.current().setNotificationCategories([alarmCategory])
  141. }
  142. //通知設定
  143. func setNotificationC(day:String, repeats:Bool){
  144. let content = UNMutableNotificationContent()
  145. content.title = alarmTime.label
  146. content.sound = UNNotificationSound.default
  147. content.categoryIdentifier = "alarmCategory"
  148. var dateComponents = DateComponents()
  149. if !day.isEmpty {
  150. dateComponents.weekday = weekDay(day: day)
  151. }
  152. dateComponents.hour = Calendar.current.component(.hour, from: datePicker.date)
  153. dateComponents.minute = Calendar.current.component(.minute, from: datePicker.date)
  154. let trigger = UNCalendarNotificationTrigger(dateMatching: dateComponents, repeats: repeats)
  155. let request = UNNotificationRequest(identifier: alarmTime.uuidString+day, content: content, trigger: trigger)
  156.  
  157. UNUserNotificationCenter.current().add(request) { (error) in
  158. if let error = error {
  159. print(error.localizedDescription)
  160. }
  161. }
  162. alarmTime.date = datePicker.date
  163. }
  164. //アラート設定
  165. func alarmSet(){
  166. removeAlarm(identifiers: alarmTime.uuidString)
  167. let shortWeekday = DateFormatter().shortWeekdaySymbols!
  168. for i in shortWeekday {
  169. removeAlarm(identifiers: alarmTime.uuidString+i)
  170. }
  171. if alarmTime.week.isEmpty {
  172. setCategories()
  173. setNotificationC(day:"", repeats: false)
  174. }else{
  175. for i in alarmTime.week {
  176. setCategories()
  177. setNotificationC(day: i, repeats: true)
  178. }
  179. }
  180. }
  181. //アラート設定削除
  182. func removeAlarm(identifiers:String){
  183. UNUserNotificationCenter.current().removePendingNotificationRequests(withIdentifiers: [identifiers])
  184. }
  185. //曜日
  186. func weekDay(day:String) -> Int{
  187. var week = DateFormatter().weekdaySymbols!
  188. switch day {
  189. case week[0]:
  190. return 1
  191. case week[1]:
  192. return 2
  193. case week[2]:
  194. return 3
  195. case week[3]:
  196. return 4
  197. case week[4]:
  198. return 5
  199. case week[5]:
  200. return 6
  201. case week[6]:
  202. return 7
  203. default:
  204. return Int()
  205. }
  206. }
  207. //キャンセル
  208. @IBAction func cancelButton(_ sender: Any) {
  209. delegate.AlarmAddVC(alarmCancel: self)
  210. dismiss(animated: true, completion: nil)
  211. }
  212. override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
  213. switch segue.identifier {
  214. case "showRepeat":
  215. guard let nextVC:AlarmRepeatVC = segue.destination as? AlarmRepeatVC else {return}
  216. nextVC.delegate = self
  217. nextVC.selectDay = alarmTime.week
  218. case "showLabel":
  219. guard let nextVC:AlarmAddLabelVC = segue.destination as? AlarmAddLabelVC else {return}
  220. nextVC.delegate = self
  221. nextVC.text = alarmTime.label
  222. default:
  223. return
  224. }
  225. }
  226. }
  227.  
  228. extension AlarmAddVC:AlarmRepeatVCDelegate {
  229. func AlarmRepeatVC(addRepeat: AlarmRepeatVC, week: [String]) {
  230. alarmTime.week = []
  231. alarmTime.repeatLabel = ""
  232. alarmTime.week += week
  233. if alarmTime.week.count == 1 {
  234. alarmTime.repeatLabel = "Every"+alarmTime.week[0]
  235. }else if alarmTime.week.isEmpty {
  236. alarmTime.repeatLabel = "Never"
  237. }else if alarmTime.week.count == 7{
  238. alarmTime.repeatLabel = "Every day"
  239. }else{
  240. let shortWeekday = DateFormatter().shortWeekdaySymbols!
  241. for i in alarmTime.week {
  242. if alarmTime.repeatLabel != "" {
  243. alarmTime.repeatLabel += ","
  244. }
  245. alarmTime.repeatLabel += shortWeekday[weekDay(day: i)]
  246. }
  247. }
  248. tableView.reloadData()
  249. }
  250. }
  251.  
  252. extension AlarmAddVC:AlarmAddLabelDelegate {
  253. func alarmAddLabel(labelText: AlarmAddLabelVC, text: String) {
  254. alarmTime.label = text
  255. tableView.reloadData()
  256. }
  257. }
  258.  
  259. extension AlarmAddVC:AlarmSnoozeCellDelegte{
  260. func alarmSnoozeCell(swichOn: AlarmSnoozeCell, On: Bool) {
  261. alarmTime.snooze = On
  262. }
  263. }
  264.  
  265. extension AlarmAddVC:AlarmDeleteCellDelegate{
  266. func alarmDeleteCell(delete: UITableViewCell) {
  267. delegate.AlarmAddVC(alarmDelete: self,alarmTime:alarmTime)
  268. dismiss(animated: true, completion: nil)
  269. }
  270. }

AlarmAddCell

  1. import UIKit
  2.  
  3. class AlarmAddCell: UITableViewCell {
  4. @IBOutlet weak var titleLabel: UILabel!
  5. @IBOutlet weak var subTitleLabel: UILabel!
  6. }

AlarmSnoozeCell

  1. import UIKit
  2.  
  3. protocol AlarmSnoozeCellDelegte {
  4. func alarmSnoozeCell(swichOn:AlarmSnoozeCell,On:Bool)
  5. }
  6.  
  7. class AlarmSnoozeCell: UITableViewCell {
  8.  
  9. @IBOutlet weak var snoozeSwitch: UISwitch!
  10. var delegate:AlarmSnoozeCellDelegte!
  11. @IBAction func switchChanged(_ sender: UISwitch) {
  12. delegate.alarmSnoozeCell(swichOn: self, On: sender.isOn)
  13. }
  14. }

AlarmRepeatVC

  1. import UIKit
  2. protocol AlarmRepeatVCDelegate {
  3. func AlarmRepeatVC(addRepeat:AlarmRepeatVC,week:[String])
  4. }
  5. class AlarmRepeatVC: UIViewController ,UITableViewDelegate,UITableViewDataSource{
  6.  
  7. @IBOutlet weak var tableView: UITableView!
  8. var delegate:AlarmRepeatVCDelegate!
  9. var week:[String] = []
  10. var selectDay:[String] = []
  11.  
  12. override func viewDidLoad() {
  13. super.viewDidLoad()
  14. // 複数選択可にする
  15. tableView.allowsMultipleSelection = true
  16. week = DateFormatter().weekdaySymbols!
  17. }
  18. override func viewWillDisappear(_ animated: Bool) {
  19. delegate.AlarmRepeatVC(addRepeat: self, week:sortWeek(selectDays: selectDay))
  20. }
  21. func sortWeek(selectDays: [String]) -> [String]{
  22. var week = DateFormatter().weekdaySymbols!
  23. var dayDictionary: [String: Int] = [:]
  24. for i in 0...6 {
  25. dayDictionary[week[i]] = i
  26. }
  27. var daysOfWeek: [String] = selectDays
  28. daysOfWeek.sort { (dayDictionary[$0] ?? 7) < (dayDictionary[$1] ?? 7)}
  29. return daysOfWeek
  30. }
  31. func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  32. return week.count
  33. }
  34. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  35. // セルを取得する
  36. let cell: UITableViewCell = tableView.dequeueReusableCell(withIdentifier: "weekCell", for: indexPath)
  37. cell.textLabel!.text = "Every"+week[indexPath.row]
  38. cell.selectionStyle = .none
  39. for i in selectDay {
  40. if week[indexPath.row] == i {
  41. cell.accessoryType = .checkmark
  42. break
  43. }else{
  44. cell.accessoryType = .none
  45. }
  46. }
  47. return cell
  48. }
  49. func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
  50. let cell = tableView.cellForRow(at:indexPath)
  51. // チェックマークを入れる
  52. cell?.accessoryType = .checkmark
  53. selectDay.append(week[indexPath.row])
  54. }
  55. // セルの選択が外れた時に呼び出される
  56. func tableView(_ tableView: UITableView, didDeselectRowAt indexPath: IndexPath) {
  57. let cell = tableView.cellForRow(at:indexPath)
  58. // チェックマークを外す
  59. cell?.accessoryType = .none
  60. selectDay = selectDay.filter { $0 != week[indexPath.row] }
  61. }
  62. }

AlarmAddLabelVC

  1. import UIKit
  2.  
  3. protocol AlarmAddLabelDelegate {
  4. func alarmAddLabel(labelText:AlarmAddLabelVC,text:String)
  5. }
  6.  
  7. class AlarmAddLabelVC: UIViewController,UITextFieldDelegate {
  8. var text:String!
  9.  
  10. @IBOutlet weak var textField: UITextField!
  11. var delegate:AlarmAddLabelDelegate!
  12. override func viewDidLoad() {
  13. super.viewDidLoad()
  14. // テキストを全消去するボタンを表示
  15. textField.clearButtonMode = .always
  16. // 改行ボタンの種類を設定
  17. textField.returnKeyType = .done
  18. // UITextFieldを追加
  19. textField.delegate = self
  20. //キーボードを表示する
  21. textField.becomeFirstResponder()
  22. textField.text = text
  23. }
  24. override func viewDidDisappear(_ animated: Bool) {
  25. //textFieldの中身が空でない時
  26. if textField.text != "", let text = textField.text{
  27. delegate.alarmAddLabel(labelText: self, text:text)
  28. }
  29. }
  30. // 完了ボタンを押した時の処理
  31. func textFieldShouldReturn(_ textField: UITextField) -> Bool {
  32. //textFieldの中身が空でない時
  33. if textField.text != "", let text = textField.text{
  34. delegate.alarmAddLabel(labelText: self, text:text)
  35. self.navigationController?.popViewController(animated: true)
  36. }
  37. return true
  38. }
  39. }
  40.  

AlarmTimeArray

  1. import UIKit
  2.  
  3. class AlarmTimeArray: NSObject,NSCoding {
  4. var date:Date
  5. var uuidString:String
  6. var label:String
  7. var sound:Bool
  8. var snooze:Bool
  9. var onOff:Bool
  10. var repeatLabel:String
  11. var week:[String]
  12. override init() {
  13. self.date = Date()
  14. self.uuidString = UUID().uuidString
  15. self.label = "Alarm"
  16. self.sound = true
  17. self.snooze = true
  18. self.onOff = true
  19. self.week = []
  20. self.repeatLabel = "Never"
  21. }
  22. func encode(with aCoder: NSCoder) {
  23. aCoder.encode(self.date, forKey: "date")
  24. aCoder.encode(self.uuidString, forKey: "uuidString")
  25. aCoder.encode(self.label, forKey: "label")
  26. aCoder.encode(self.sound, forKey: "sound")
  27. aCoder.encode(self.snooze, forKey: "snooze")
  28. aCoder.encode(self.onOff, forKey: "onOff")
  29. aCoder.encode(self.week, forKey: "week")
  30. aCoder.encode(self.repeatLabel, forKey: "repeatLabel")
  31.  
  32. }
  33. required init?(coder aDecoder: NSCoder) {
  34. date = aDecoder.decodeObject(forKey: "date") as! Date
  35. uuidString = aDecoder.decodeObject(forKey: "uuidString") as! String
  36. label = aDecoder.decodeObject(forKey: "label") as! String
  37. sound = aDecoder.decodeBool(forKey: "sound")
  38. snooze = aDecoder.decodeBool(forKey: "snooze")
  39. onOff = aDecoder.decodeBool(forKey: "onOff")
  40. week = aDecoder.decodeObject(forKey: "week") as! [String]
  41. repeatLabel = aDecoder.decodeObject(forKey: "repeatLabel") as! String
  42. }
  43. }
githubはこちら

参考

UITableViewの編集モードを使ってCellの削除を実装するまで
UITableViewのセルを選択不可にする方法
【Swift4】配列の中身(数値・文字・日付)を比較してソートする方法【Xcode9】
UITableViewでセルを複数選択する
【Swift】UserDefaultsに自作クラスのデータを保存する方法(iOS12対応)[Swift][Obj-C]UUIDStringの使い分け

QooQ