This repository was archived by the owner on Apr 23, 2026. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 51
Expand file tree
/
Copy pathSessionsTableViewController.swift
More file actions
245 lines (201 loc) · 9.14 KB
/
SessionsTableViewController.swift
File metadata and controls
245 lines (201 loc) · 9.14 KB
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
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
//
// SessionsTableViewController.swift
// trySwift
//
// Created by Natasha Murashev on 2/10/16.
// Copyright © 2016 NatashaTheRobot. All rights reserved.
//
import UIKit
import XLPagerTabStrip
import TrySwiftData
class SessionsTableViewController: UITableViewController {
private lazy var needsToScrollToCurrentSession = Calendar.current.isDateInToday(conferenceDay.date)
var conferenceDay: ConferenceDay
weak var scheduleViewController: ScheduleViewController?
fileprivate let sessionDetailsSegue = "sessionDetailsSegue"
fileprivate var didShowDetail = false
init(conferenceDay: ConferenceDay, scheduleViewController: ScheduleViewController) {
self.conferenceDay = conferenceDay
self.scheduleViewController = scheduleViewController
super.init(style: .plain)
}
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func viewDidLoad() {
super.viewDidLoad()
configureTableView()
if traitCollection.forceTouchCapability == .available {
registerForPreviewing(with: self, sourceView: tableView)
}
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
if needsToScrollToCurrentSession {
needsToScrollToCurrentSession = false
scrollToCurrentSession(animated: false)
}
guard
let firstSelectableSession = conferenceDay.sessionBlocks
.flatMap({ $0.sessions })
.filter({ SessionViewModel(session: $0).selectable }).first,
let firstSelectableSessionVC = viewController(for: firstSelectableSession),
let isCollapsed = splitViewController?.isCollapsed,
!isCollapsed,
!didShowDetail else { return }
didShowDetail = true
scheduleViewController?.performSegue(withIdentifier: sessionDetailsSegue, sender: firstSelectableSessionVC)
}
}
// MARK: - Table view data source
extension SessionsTableViewController {
override func numberOfSections(in tableView: UITableView) -> Int {
return conferenceDay.sessionBlocks.count
}
override func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
return conferenceDay.sessionBlocks[section].sessions.count
}
override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
let cell = tableView.dequeueReusableCell(forIndexPath: indexPath) as SessionTableViewCell
let session = conferenceDay.sessionBlocks[indexPath.section].sessions[indexPath.row]
cell.configure(withSession: session)
return cell
}
override func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? {
let session = conferenceDay.sessionBlocks[section]
let sessionDateFormatter = DateFormatter.sessionDateFormatter
let startString = sessionDateFormatter.string(from: session.startTime)
let endString = sessionDateFormatter.string(from: session.endTime)
return "\(startString) - \(endString)"
}
}
// MARK: - Table view delegate
extension SessionsTableViewController {
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
tableView.deselectRow(at: indexPath, animated: true)
let session = conferenceDay.sessionBlocks[indexPath.section].sessions[indexPath.row]
guard let viewController = viewController(for: session) else { return }
scheduleViewController?.performSegue(withIdentifier: sessionDetailsSegue, sender: viewController)
}
}
extension SessionsTableViewController: IndicatorInfoProvider {
public func indicatorInfo(for pagerTabStripController: PagerTabStripViewController) -> IndicatorInfo {
return IndicatorInfo(title: DateFormatter.dayDateFormatter.string(from: conferenceDay.date))
}
}
extension SessionsTableViewController: UIViewControllerPreviewingDelegate {
func previewingContext(_ previewingContext: UIViewControllerPreviewing, viewControllerForLocation location: CGPoint) -> UIViewController? {
guard let indexPath = tableView.indexPathForRow(at: location) else { return nil }
// This will show the cell clearly and blur the rest of the screen for our peek.
previewingContext.sourceRect = tableView.rectForRow(at: indexPath)
let session = conferenceDay.sessionBlocks[indexPath.section].sessions[indexPath.row]
return viewController(for: session)
}
func previewingContext(_ previewingContext: UIViewControllerPreviewing, commit viewControllerToCommit: UIViewController) {
scheduleViewController?.performSegue(withIdentifier: sessionDetailsSegue, sender: viewControllerToCommit)
}
}
extension SessionsTableViewController {
func configureTableView() {
tableView.register(SessionTableViewCell.self)
tableView.estimatedRowHeight = 160
tableView.rowHeight = UITableViewAutomaticDimension
}
}
private extension SessionsTableViewController {
func viewController(for session: Session) -> UIViewController? {
switch session.type {
case .talk, .lightningTalk:
if let presentation = session.presentation {
return sessionDetails(presentation, session: session)
}
case .officeHours:
if let speaker = session.presentation?.speaker {
return officeHourDetails(speaker, session: session)
}
case .workshop:
if let presentation = session.presentation, let venue = session.venue {
return workshopDetails(presentation, session: session, venue: venue)
}
else if let presentation = session.presentation {
return sessionDetails(presentation, session: session)
}
else if let event = session.event {
return webDisplay(event)
}
case .meetup:
if let event = session.event {
return webDisplay(event)
}
case .coffeeBreak:
if let sponsor = session.sponsor {
return webDisplay(sponsor)
}
case .sponsoredDemo:
if let sponsor = session.sponsor {
return webDisplay(sponsor)
}
case .party:
if let venue = session.venue {
return venueDetails(venue)
}
default:
return nil
}
return nil
}
func sessionDetails(_ presentation: Presentation, session: Session) -> UIViewController {
let storyboard = UIStoryboard(name: "Main", bundle: nil)
let sessionDetailsVC = storyboard.instantiateViewController(withIdentifier: String(describing: SessionDetailsViewController.self)) as! SessionDetailsViewController
sessionDetailsVC.session = session
sessionDetailsVC.presentation = presentation
return sessionDetailsVC
}
func officeHourDetails(_ speaker: Speaker, session: Session) -> UIViewController {
let officeHoursVC = OfficeHoursDetailViewController()
officeHoursVC.speaker = speaker
officeHoursVC.session = session
return officeHoursVC
}
func webDisplay(_ event: Event) -> UIViewController {
let webViewController = WebDisplayViewController()
webViewController.url = URL(string: event.website!)
webViewController.displayTitle = event.title
return webViewController
}
func webDisplay(_ sponsor: Sponsor) -> UIViewController {
let webViewController = WebDisplayViewController()
webViewController.url = URL(string: sponsor.url!)
webViewController.displayTitle = sponsor.name
return webViewController
}
func venueDetails(_ venue: Venue) -> UIViewController {
let venueDetailsVC = VenueTableViewController(venue: venue)
venueDetailsVC.tableView.contentInset = UIEdgeInsets(top: 80,left: 0,bottom: 0,right: 0)
return venueDetailsVC
}
func workshopDetails(_ presentation: Presentation, session: Session, venue: Venue) -> UIViewController {
let storyboard = UIStoryboard(name: "Main", bundle: nil)
let workshopDetailVC = storyboard.instantiateViewController(withIdentifier: String(describing: WorkshopDetailViewController.self)) as! WorkshopDetailViewController
workshopDetailVC.session = session
workshopDetailVC.presentation = presentation
workshopDetailVC.venue = venue
return workshopDetailVC
}
}
extension SessionsTableViewController {
func scrollToCurrentSession(animated: Bool) {
let secondsFromGMT = TimeZone.current.secondsFromGMT()
guard
let date = Date().changed(second: secondsFromGMT),
let section = conferenceDay.sessionBlocks.index(where: { date < $0.endTime }),
!conferenceDay.sessionBlocks[section].sessions.isEmpty
else { return }
tableView.scrollToRow(at: IndexPath(row: 0, section: section), at: .top, animated: animated)
}
}
extension SessionsTableViewController: ScrollableToTop {
func scrollAfterTabTap() {
scrollToCurrentSession(animated: true)
}
}