Files
yattee/Yattee/Views/Toast/ToastOverlayView.swift
Arkadiusz Fal b163864628 Add interactive swipe-to-dismiss for iOS toasts
Toast cards now follow the finger upward and dismiss on either a
sufficient drag or a fast flick (via predicted-end translation). The
auto-dismiss timer pauses while the user is dragging and re-arms if
they release without dismissing.
2026-05-08 03:04:32 +02:00

94 lines
2.4 KiB
Swift

//
// ToastOverlayView.swift
// Yattee
//
// Overlay container for displaying toast notifications at the top of the screen.
//
import SwiftUI
/// Overlay that displays toast notifications at the top of the screen.
struct ToastOverlayView: View {
@Environment(\.appEnvironment) private var appEnvironment
/// The scope of toasts to display in this overlay.
let scope: ToastScope
private var toastManager: ToastManager? {
appEnvironment?.toastManager
}
/// Toasts filtered by scope
private var scopedToasts: [Toast] {
toastManager?.toasts(for: scope) ?? []
}
var body: some View {
VStack(spacing: 8) {
ForEach(scopedToasts) { toast in
ToastCardView(
toast: toast,
onDismiss: {
toastManager?.dismiss(id: toast.id)
},
onAction: toast.action?.handler,
onDragBegan: {
toastManager?.pauseAutoDismiss(id: toast.id)
},
onDragCancelled: {
toastManager?.resumeAutoDismiss(id: toast.id)
}
)
.transition(.move(edge: .top).combined(with: .opacity))
}
}
.padding(.top, topPadding)
.padding(.horizontal, 16)
.animation(.spring(response: 0.3, dampingFraction: 0.7), value: scopedToasts.count)
}
private var topPadding: CGFloat {
#if os(iOS)
// In player scope (sheet), only need minimal padding since safe area is handled by sheet
// In main scope, account for Dynamic Island / notch
return scope == .player ? 8 : 60
#elseif os(macOS)
return 16
#elseif os(tvOS)
// TV safe area
return 60
#endif
}
}
// MARK: - View Extension
extension View {
/// Adds a toast overlay for the main window scope.
func toastOverlay() -> some View {
overlay(alignment: .top) {
ToastOverlayView(scope: .main)
}
}
/// Adds a toast overlay for the player scope.
func playerToastOverlay() -> some View {
overlay(alignment: .top) {
ToastOverlayView(scope: .player)
}
}
}
// MARK: - Preview
#Preview {
ZStack {
Color.gray.opacity(0.3)
.ignoresSafeArea()
Text("Main Content")
}
.toastOverlay()
.appEnvironment(.preview)
}