Disabling pop-up presentations in a WKWebView in an immersive app

Hello, I am creating a VisionOS application where a simple web browser is implemented in an immersive view by using WKWebView. Presentations cause a crash with "Presentations are not permitted within volumetric window scenes."

Some potential ways to fix this:

Disable presentations entirely, and have websites use a fallback.

Allow me to create handlers for when presentations would be created and do something else.

Allow presentations to be created as Attachments, which is how the WKWebView is implemented.

I have looked at the WKWebView documentation, but haven't found a way to attempt these.

Relevant code:

import SwiftUI
import WebKit

struct WebView: UIViewRepresentable {
    typealias UIViewType = WKWebView
    
    let webView: WKWebView
    
    func makeUIView(context: Context) -> WKWebView {
        return webView
    }
    
    func updateUIView(_ uiView: WKWebView, context: Context) {
        
    }
}
import SwiftUI
import WebKit


class WebViewModel: ObservableObject {
    let webView: WKWebView
    let url: URL
    
    @Published var canGoBack: Bool = false
    @Published var canGoForward: Bool = false
    
    @Published var urlString: String = ""
    
    init() {
        let config = WKWebViewConfiguration()
        webView = WKWebView(frame: .zero, configuration: config)
        url = URL(string: "https://www.google.com")!
        setupBindings()
        
        loadURL()
    }
    
    private func setupBindings() {
        webView.publisher(for: \.canGoBack)
            .assign(to: &$canGoBack)
        
        webView.publisher(for: \.canGoForward)
            .assign(to: &$canGoForward)
    }
    
    func loadURL() {
        
        let urlElements = urlString.split(separator: ".")
        print(urlElements.count)
        if (urlElements.count > 1 && !urlString.contains(" ")) {
            f (!urlString.lowercased().contains("https://")) {
                urlString = String("https://").appending(urlString)
            }
        } else {
            urlString = "https://www.google.com/search?q=" + urlString.replacingOccurrences(of: " ", with: "+")
        }
        
        guard let url = URL(string: urlString) else {
            return
        }
        
        webView.load(URLRequest(url: url))
    }
    
    func goForward() {
        webView.goForward()
    }
    func goBack() {
        webView.goBack()
    }
}
import SwiftUI

struct BrowserView: View {
    
    @StateObject var model: WebViewModel
    
    var body: some View {
        ZStack(alignment: .bottom) {
            Color.black
                .ignoresSafeArea()
            
            VStack(spacing: 0) {
                HStack(spacing: 10) {
                    HStack {
                        TextField("Enter an url", text: $model.urlString)
                            .keyboardType(.URL)
                            .textInputAutocapitalization(.none)
                            .padding(10)
                            .onKeyPress(.return) {
                                model.loadURL()
                                return .handled
                            }
                        Spacer()
                    }
                    .background(Color.gray)
                    .presentationCornerRadius(30)
                    
                    Button("Go", action: {
                        model.loadURL()
                    })
                    .padding(10)
                }
                .padding(10)
                
                WebView(webView: model.webView)
                
                HStack(spacing: 10) {
                    Button(action: {
                        model.goBack()
                    }, label: {
                        Image(systemName: "arrowshape.turn.up.backward")
                    })
                    .disabled(!model.canGoBack)
                    
                    Button(action: {
                        model.goForward()
                    }, label: {
                        Image(systemName: "arrowshape.turn.up.right")
                    })
                    .disabled(!model.canGoForward)
                    
                    Spacer()
                }
                .padding(10)
            }
        }
    }
}

Hi @zlys

Have you considered opening the browser in a window instead of a volume?

Presentations are supported in a window (e.g. a WindowGroup with a windowStyle of .plain or .automatic) which you can open from within an immersive space using the openWindow environment function. Presenting windows and spaces contains relevant information..

Hi @Vision Pro Engineer , thank you for your answer. The reason the browser window is opened in a volume is that the concept of the application is to have the element move alongside the user as he moves around. We did not find a way to accomplish this with WindowGroups, which seem designed to remain static in the 3D space.

Disabling pop-up presentations in a WKWebView in an immersive app
 
 
Q