passepartout-apple/Passepartout/Library/Sources/UILibrary/Views/Paywall/PaywallModifier.swift
Davide 2c1ccbcbfd
Finalize paywall UI (#831)
- Use StoreKit views when available
- Offer one-time purchase
- Recurring subscriptions for all features
- Restore purchases

Remove .siri (Shortcuts), now free.

Closes #819
Closes #469
2024-11-07 18:27:36 +01:00

107 lines
3.0 KiB
Swift

//
// PaywallModifier.swift
// Passepartout
//
// Created by Davide De Rosa on 9/11/24.
// Copyright (c) 2024 Davide De Rosa. All rights reserved.
//
// https://github.com/passepartoutvpn
//
// This file is part of Passepartout.
//
// Passepartout is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// Passepartout is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Passepartout. If not, see <http://www.gnu.org/licenses/>.
//
import CommonLibrary
import SwiftUI
public struct PaywallModifier: ViewModifier {
@Binding
private var reason: PaywallReason?
@State
private var isPresentingRestricted = false
@State
private var paywallArguments: PaywallArguments?
public init(reason: Binding<PaywallReason?>) {
_reason = reason
}
public func body(content: Content) -> some View {
content
.alert(
Strings.Alerts.Iap.Restricted.title,
isPresented: $isPresentingRestricted,
actions: {
Button(Strings.Global.ok) {
reason = nil
isPresentingRestricted = false
}
},
message: {
Text(Strings.Alerts.Iap.Restricted.message)
}
)
.themeModal(item: $paywallArguments) { args in
NavigationStack {
PaywallView(
isPresented: isPresentingPurchase,
feature: args.feature,
suggestedProduct: args.product
)
}
.frame(idealHeight: 400)
}
.onChange(of: reason) {
switch $0 {
case .restricted:
isPresentingRestricted = true
case .purchase(let feature, let product):
paywallArguments = PaywallArguments(feature: feature, product: product)
default:
break
}
}
}
}
private extension PaywallModifier {
var isPresentingPurchase: Binding<Bool> {
Binding {
paywallArguments != nil
} set: {
if !$0 {
// make sure to reset this to allow paywall to appear again
reason = nil
paywallArguments = nil
}
}
}
}
private struct PaywallArguments: Identifiable {
let feature: AppFeature
let product: AppProduct?
var id: String {
feature.id
}
}