SwiftUI how to build multiple navigation links

333 Views Asked by At

I'm using a custom NavigationButton from this answer to be able to load data for a specific view when navigating to a new view from a list. It works great, but the problem is when I attempt to use it inside a ForEach loop I end up with the multiple navigation links with multiple isActive @State variables that are all being told to push a new view, which causes SwiftUI to navigate to the new view then automatically back to parent view (bug). How can I preserve the functionality of being able to fire code to run while navigating to a new view without triggering the bug.

ForEach(workouts) { workout in
    NavigationButton(
        action: {
            //load data for Destination
        },
        destination: {
            WorkoutDetailView().environmentObject(dataStore)
        },
        workoutRow: { WorkoutRow(trackerWorkout: workout) }
    )
    
}

struct NavigationButton<Destination: View, WorkoutRow: View>: View {
    var action: () -> Void = { }
    var destination: () -> Destination
    var workoutRow: () -> WorkoutRow
    
    @State private var isActive: Bool = false
    
    var body: some View {
        Button(action: {
            self.action()
            self.isActive.toggle()
        }) {
            workoutRow()
                .background(
                    ScrollView { // Fixes a bug where the navigation bar may become hidden on the pushed view
                        NavigationLink(destination: LazyDestination { self.destination() },
                                       isActive: self.$isActive) { EmptyView() }
                    }
                )
        }
    }
}

// This view lets us avoid instantiating our Destination before it has been pushed.
struct LazyDestination<Destination: View>: View {
    var destination: () -> Destination
    var body: some View {
        self.destination()
    }
}
0

There are 0 best solutions below