Skip to content

Commit

Permalink
feat: reset nearby transit scroll position
Browse files Browse the repository at this point in the history
  • Loading branch information
Brandon Rodriguez authored and Brandon Rodriguez committed Apr 11, 2024
1 parent 671b0a8 commit 3d04cd1
Show file tree
Hide file tree
Showing 3 changed files with 176 additions and 82 deletions.
100 changes: 73 additions & 27 deletions iosApp/iosApp/Pages/NearbyTransit/NearbyTransitView.swift
Original file line number Diff line number Diff line change
Expand Up @@ -21,31 +21,27 @@ struct NearbyTransitView: View {
@ObservedObject var scheduleFetcher: ScheduleFetcher
@ObservedObject var predictionsFetcher: PredictionsFetcher
@ObservedObject var alertsFetcher: AlertsFetcher
@State var nearbyWithRealtimeInfo: [StopAssociatedRoute]?
@State var now = Date.now
@State var scrollPosition: String?

let timer = Timer.publish(every: 5, on: .main, in: .common).autoconnect()
let inspection = Inspection<Self>()

var body: some View {
VStack {
if let nearby = nearbyFetcher.withRealtimeInfo(
schedules: scheduleFetcher.schedules,
predictions: predictionsFetcher.predictions,
alerts: alertsFetcher.alerts,
filterAtTime: now.toKotlinInstant()
) {
List(nearby, id: \.route.id) { nearbyRoute in
NearbyRouteView(nearbyRoute: nearbyRoute, now: now.toKotlinInstant())
}.putAboveWhen(predictionsFetcher.errorText) { errorText in
IconCard(iconName: "network.slash", details: errorText)
}
if let nearbyWithRealtimeInfo {
nearbyList(nearbyWithRealtimeInfo)
} else {
Text("Loading...")
.frame(maxWidth: .infinity)
.padding(.top, 24)
}
}
.onAppear {
getNearby(location: locationProvider.location)
joinPredictions()
updateNearbyRoutes()
didAppear?(self)
}
.onChange(of: globalFetcher.response) { _ in
Expand All @@ -55,31 +51,59 @@ struct NearbyTransitView: View {
getNearby(location: newLocation)
}
.onChange(of: nearbyFetcher.nearbyByRouteAndStop) { _ in
updateNearbyRoutes()
getSchedule()
joinPredictions()
scrollToTop()
}
.onChange(of: scheduleFetcher.schedules) { _ in
updateNearbyRoutes()
}
.onChange(of: predictionsFetcher.predictions) { _ in
updateNearbyRoutes()
}
.onChange(of: alertsFetcher.alerts) { _ in
updateNearbyRoutes()
}
.onChange(of: scenePhase) { newPhase in
if newPhase == .inactive {
leavePredictions()
} else if newPhase == .active {
joinPredictions()
} else if newPhase == .background {
leavePredictions()
}
onSceneChange(newPhase)
}
.onReceive(timer) { input in
now = input
updateNearbyRoutes()
}
.onReceive(inspection.notice) { inspection.visit(self, $0) }
.onDisappear {
leavePredictions()
}
.replaceWhen(nearbyFetcher.errorText) { errorText in
IconCard(iconName: "network.slash", details: errorText)
.refreshable(nearbyFetcher.loading) { getNearby(location: locationProvider.location) }
errorCard(errorText)
}
}

private func nearbyList(_ routes: [StopAssociatedRoute]) -> some View {
ScrollViewReader { proxy in
List(routes, id: \.route.id) { nearbyRoute in
NearbyRouteView(nearbyRoute: nearbyRoute, now: now.toKotlinInstant())
}
.onChange(of: scrollPosition) { id in
guard let id else { return }
withAnimation {
proxy.scrollTo(id, anchor: .center)
scrollPosition = nil
}
}
.putAboveWhen(predictionsFetcher.errorText) { errorText in
IconCard(iconName: "network.slash", details: errorText)
}
}
}

private func errorCard(_ errorText: Text) -> some View {
IconCard(iconName: "network.slash", details: errorText)
.refreshable(nearbyFetcher.loading) { getNearby(location: locationProvider.location) }
}

var didAppear: ((Self) -> Void)?

func getNearby(location: CLLocationCoordinate2D) {
Expand All @@ -99,17 +123,39 @@ struct NearbyTransitView: View {
}

func joinPredictions() {
Task {
guard let stopIds = nearbyFetcher.nearbyByRouteAndStop?
.stopIds() else { return }
let stopIdList = Array(stopIds)
predictionsFetcher.run(stopIds: stopIdList)
}
guard let stopIds = nearbyFetcher.nearbyByRouteAndStop?
.stopIds() else { return }
let stopIdList = Array(stopIds)
predictionsFetcher.run(stopIds: stopIdList)
}

func leavePredictions() {
predictionsFetcher.leave()
}

private func scrollToTop() {
guard let id = nearbyWithRealtimeInfo?.first?.route.id else { return }
scrollPosition = id
}

private func updateNearbyRoutes() {
nearbyWithRealtimeInfo = nearbyFetcher.withRealtimeInfo(
schedules: scheduleFetcher.schedules,
predictions: predictionsFetcher.predictions,
alerts: alertsFetcher.alerts,
filterAtTime: now.toKotlinInstant()
)
}

private func onSceneChange(_ phase: ScenePhase) {
Task {
predictionsFetcher.leave()
if phase == .inactive {
leavePredictions()
} else if phase == .active {
joinPredictions()
} else if phase == .background {
leavePredictions()
}
}
}
}
Expand Down
1 change: 1 addition & 0 deletions iosApp/iosApp/Utils/Backport/PartialSheetModifier.swift
Original file line number Diff line number Diff line change
Expand Up @@ -106,6 +106,7 @@ private extension PartialSheetRepresentable {
controller.animateChanges {
controller.detents = detents.map(\.uiKitDetent)
controller.prefersScrollingExpandsWhenScrolledToEdge = true
controller.prefersGrabberVisible = true

if let largestUndimmedDetent {
controller.largestUndimmedDetentIdentifier = .init(largestUndimmedDetent.rawValue)
Expand Down
Loading

0 comments on commit 3d04cd1

Please sign in to comment.