Issue with SwiftUI NavigationStack, Searchable Modifier, and Returning to Root View

I'm facing an issue with SwiftUI's NavigationStack when using the searchable modifier. Everything works as expected when navigating between views, but if I use the search bar to filter a list and then tap on a filtered result, I can navigate to the next view. However, in the subsequent view, my "Set and Return to Root" button, which is supposed to call popToRoot(), does not work. Here's the setup:

Structure: RootView: Contains a list with items 1-7. ActivityView: Contains a list of activities that can be filtered with the searchable modifier. SettingView: Contains a button labeled "Set and Return to Root" that calls popToRoot() to navigate back to the root view.

RootView

struct RootView: View {
    @EnvironmentObject var navManager: NavigationStateManager
    
    var body: some View {
        NavigationStack(path: $navManager.selectionPath) {
            List(1...7, id: \.self) { item in
                Button("Element \(item)") {
                    // Navigate to ActivityView with an example string
                    navManager.selectionPath.append(NavigationTarget.activity)
                }
            }
            .navigationTitle("Root View")
            .navigationDestination(for: NavigationTarget.self) { destination in
                switch destination {
                case .activity:
                    ActivityView()
                case .settings:
                    SettingsView()
                }
            }
        }
    }
}

ActivityView

struct ActivityView: View {
    @EnvironmentObject var navManager: NavigationStateManager
    
    let activities = ["Running", "Swimming", "Cycling", "Hiking", "Yoga", "Weightlifting", "Boxing"]
    
    @State private var searchText = ""
    
    var filteredActivities: [String] {
        if searchText.isEmpty {
            return activities
        } else {
            return activities.filter { $0.localizedCaseInsensitiveContains(searchText) }
        }
    }
    
    var body: some View {
        List {
            ForEach(filteredActivities, id: \.self) { activity in
                NavigationLink(
                    destination: SettingsView(), // Navigiere zur SettingsView
                    label: {
                        HStack {
                            Text(activity)
                                .padding()
                            Spacer()
                        }
                    }
                )
            }
        }
        
        .navigationTitle("Choose Activity")
        .searchable(text: $searchText, placement: .navigationBarDrawer(displayMode: .always), prompt: "Search Activities")
    }
}

SettingView

struct SettingsView: View {
    @EnvironmentObject var navManager: NavigationStateManager

    var body: some View {
        VStack {
            Text("Settings")
                .font(.largeTitle)
                .padding()

            Button("Set and Return to Root") {
                // Pop to the root view when the button is pressed
                navManager.popToRoot()
            }
            .padding()
            .background(Color.blue)
            .foregroundColor(.white)
            .cornerRadius(10)
        }
        .navigationTitle("Settings")
    }
}

NavigationStateManager

// Define enum globally at the top
enum NavigationTarget {
    case activity
    case settings
}

class NavigationStateManager: ObservableObject {
    @Published var selectionPath = NavigationPath()

    func popToRoot() {
        selectionPath = NavigationPath()
    }

    func popView() {
        selectionPath.removeLast()
    }
}

Problem

When I search in the ActivityView and tap on a filtered result, I successfully navigate to the SettingView. However, in this view, pressing the "Set and Return to Root" button does not trigger the navigation back to RootView, even though popToRoot() is being called.

This issue only occurs when using the search bar and filtering results. If I navigate without using the search bar, the button works as expected.

Question

Why is the popToRoot() function failing after a search operation, and how can I ensure that I can return to the root view after filtering the list?

Any insights or suggestions would be greatly appreciated!

The issue only occurs with iOS 18, but not with iOS 17.

@zikomiko This seems like a bug to me. I'd appreciate it if you could open a bug report, include the code snippet that reproduces the issue, the OS version and the steps to reproduce the issue. Please post the FB number here once you do.

If you have any questions about filing a bug report, take a look at Bug Reporting: How and Why?

Yes, by now I'm fairly sure it's a bug. I was able to reproduce the issue in all beta versions of iOS 18. The error is not present in the iOS 17 line. I've submitted a report in the Feedback Assistant with a sample Xcode project. FB15221588

I have the exact same issue.

I'm having this issue on the app I'm developing, one way that I found to fix it was wrapping the view that had the searchable with a NavigationView

So I've made this small playground code that cannot pop to root when search is active.

import SwiftUI
import PlaygroundSupport

struct RootView: View {
    @State private var path = NavigationPath()

    var body: some View {
        NavigationStack(path: $path) {
            DetailView(
                number: 0,
                popToRoot: { path = NavigationPath() }
            )
                .navigationDestination(for: Int.self) { i in
                    DetailView(
                        number: i,
                        popToRoot: { path = NavigationPath()  }
                    )
                }
        }
    }
}

struct DetailView: View {
    var number: Int
    var popToRoot: () -> Void
    @State var search = ""

    var body: some View {
        VStack {
            NavigationLink("Go to Random Number", value: Int.random(in: 1...1000))
            Button("Go to root") {
                popToRoot()
            }
        }
        .navigationTitle("Number: \(number)")
        .searchable(text: $search, prompt: "Search")
    }
}

PlaygroundPage.current.needsIndefiniteExecution = true
PlaygroundPage.current.setLiveView(RootView())

When I wrap it with NavigationView I can activate the search without having the pop to root issue.

import SwiftUI
import PlaygroundSupport

struct RootView: View {
    @State private var path = NavigationPath()

    var body: some View {
        NavigationStack(path: $path) {
            DetailView(
                number: 0,
                pop: nil,
                popToRoot: { path = NavigationPath() }
            )
                .navigationDestination(for: Int.self) { i in
                    DetailView(
                        number: i,
                        pop: { path.removeLast() },
                        popToRoot: { path = NavigationPath()  }
                    )
                }
        }
    }
}

struct DetailView: View {
    var number: Int
    var pop: (() -> Void)?
    var popToRoot: () -> Void
    @State var search = ""

    var body: some View {
        // Add this NavigationView with navigation bar hidden
        NavigationView {
            VStack {
                NavigationLink("Go to Random Number", value: Int.random(in: 1...1000))
                Button("Go to root") {
                    popToRoot()
                }
            }
            .toolbar {
                // Programatically add a back button since navigation view bar is hidden
                if pop != nil {
                    ToolbarItem(placement: .navigation) {
                        Button("Back", systemImage: "chevron.backward") {
                            pop?()
                        }
                        .buttonStyle(.borderless)
                        .labelStyle(.titleAndIcon)
                        .fixedSize()
                    }
                }
            }
            .navigationTitle("Number: \(number)")
            .searchable(text: $search, prompt: "Search")
        }.navigationBarHidden(true)
    }
}

PlaygroundPage.current.needsIndefiniteExecution = true
PlaygroundPage.current.setLiveView(RootView())

It definitely shouldn't be the way to fix this, but it's an okayish workaround to fix it for now...

Issue with SwiftUI NavigationStack, Searchable Modifier, and Returning to Root View
 
 
Q