Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
111 changes: 111 additions & 0 deletions Sources/SwiftNextcloudUI/Modifiers/LoginSheet.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,111 @@
// SPDX-FileCopyrightText: Nextcloud GmbH
// SPDX-FileCopyrightText: 2025 tdhooghe
// SPDX-License-Identifier: GPL-3.0-or-later

import SwiftUI

#if os(iOS)
import AuthenticationServices
#endif

///
/// Presents the Nextcloud Login Flow v2 authentication UI.
///
/// On **iOS**, this uses `ASWebAuthenticationSession` which opens a system browser sheet
/// that properly handles cross-domain OIDC redirects, passkeys, and deep links.
///
/// On **macOS**, this presents a ``WebView`` in a sheet.
///
/// Credentials are obtained via the host app's polling mechanism on both platforms.
///
struct LoginSheet: ViewModifier {
let userAgent: String?
let onDismiss: () -> Void

@Binding var loginURL: URL?
@Binding var isPresented: Bool

#if os(iOS)
@State private var authSession: ASWebAuthenticationSession?
@State private var sessionCoordinator = SessionCoordinator()
#endif

init(loginURL: Binding<URL?>, isPresented: Binding<Bool>, userAgent: String?, onDismiss: @escaping () -> Void) {
self._loginURL = loginURL
self._isPresented = isPresented
self.userAgent = userAgent
self.onDismiss = onDismiss
}

func body(content: Content) -> some View {
content
#if os(macOS)
.sheet(isPresented: $isPresented, onDismiss: onDismiss) {
WebView(initialURL: $loginURL, userAgent: userAgent)
.ignoresSafeArea()
.frame(minWidth: 800, minHeight: 800)
}
#else
.onChange(of: isPresented) { _, presented in
if presented, let url = loginURL {
startAuthSession(url: url)
} else {
authSession?.cancel()
authSession = nil
}
}
#endif
}

#if os(iOS)

private func startAuthSession(url: URL) {
let session = ASWebAuthenticationSession(url: url, callbackURLScheme: "nc") { _, error in
authSession = nil
if let error = error as? ASWebAuthenticationSessionError, error.code == .canceledLogin {
onDismiss()
}
}

session.presentationContextProvider = sessionCoordinator
session.prefersEphemeralWebBrowserSession = true
authSession = session
session.start()
}

#endif
}

extension View {
///
/// Present the login authentication UI appropriate for the current platform.
///
/// See ``LoginSheet`` for the implementation.
///
func loginSheet(loginURL: Binding<URL?>, isPresented: Binding<Bool>, userAgent: String?, onDismiss: @escaping () -> Void) -> some View {
modifier(LoginSheet(loginURL: loginURL, isPresented: isPresented, userAgent: userAgent, onDismiss: onDismiss))
}
}

// MARK: - ASWebAuthenticationSession Coordinator

#if os(iOS)

///
/// Provides the presentation anchor for `ASWebAuthenticationSession` in SwiftUI contexts.
///
@MainActor
private class SessionCoordinator: NSObject, ASWebAuthenticationPresentationContextProviding {
nonisolated func presentationAnchor(for session: ASWebAuthenticationSession) -> ASPresentationAnchor {
MainActor.assumeIsolated {
guard let windowScene = UIApplication.shared.connectedScenes.first(where: { $0.activationState == .foregroundActive }) as? UIWindowScene,
let window = windowScene.windows.first(where: \.isKeyWindow)
else {
return ASPresentationAnchor()
}
return window
}
}
}

#endif
8 changes: 4 additions & 4 deletions Sources/SwiftNextcloudUI/Views/ServerAddressView.swift
Original file line number Diff line number Diff line change
Expand Up @@ -216,7 +216,7 @@ public struct ServerAddressView: View, QRCodeParsing, URLSanitizing {
.safeAreaPadding(.all)
}
.ignoresSafeArea()
.webSheet(initialURL: $loginAddress, isPresented: $isPresentingWebView, userAgent: userAgent, onDismiss: endWebView)
.loginSheet(loginURL: $loginAddress, isPresented: $isPresentingWebView, userAgent: userAgent, onDismiss: endWebView)
.alert(String(localized: "Login Failed", comment: "Alert title"), isPresented: $isPresentingAlert) {
Button(role: .cancel) {
errorMessage = nil
Expand Down Expand Up @@ -294,12 +294,12 @@ public struct ServerAddressView: View, QRCodeParsing, URLSanitizing {
}

///
/// Dismisses the sheet with the web view.
/// Dismisses the login UI.
///
/// Multiple paths can lead to here. In example:
///
/// - The user dismisses the sheet without logging in.
/// - The login completed successfully.
/// - The user dismisses the session without logging in.
/// - The login completed successfully (detected by polling).
///
func endWebView() {
if let pollingToken {
Expand Down