passepartout-apple/Passepartout/Library/Sources/AppUIMain/Views/Profile/iOS/ProfileEditView+iOS.swift
Davide e07833b2a4
Revisit in-app eligibility for iCloud sharing (#837)
Restore .sharing feature:

- Merge "Apple TV" into "iCloud" section
  - "Enabled", disabled if ineligible for .sharing
  - "Apple TV", disabled if ineligible for .appleTV || !isShared
- Footer about TV restrictions

Paywalls:

- "Share on iCloud" if ineligible for .sharing
- "Drop TV restriction" if eligible for .sharing but not for .appleTV
  - Applies to full version products (user level 2)
  - Suggest Apple TV product

Restrictions:

- Toggle CloudKit sync on remote repository based on .sharing
eligibility
- Do not start tunnel on Apple TV if ineligible for .appleTV

Fixes:

- Incorrect zip() publishers in remote repository
- Resolve duplicates in Core Data, first profile wins sorted by
lastUpdate descending
- Reload receipt on OOB IAPManager events
2024-11-09 15:20:59 +01:00

188 lines
5.3 KiB
Swift

//
// ProfileEditView+iOS.swift
// Passepartout
//
// Created by Davide De Rosa on 6/22/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/>.
//
#if os(iOS)
import CommonLibrary
import CommonUtils
import PassepartoutKit
import SwiftUI
struct ProfileEditView: View, Routable {
@ObservedObject
var profileEditor: ProfileEditor
let moduleViewFactory: any ModuleViewFactory
@Binding
var path: NavigationPath
var flow: ProfileCoordinator.Flow?
@State
private var malformedModuleIds: [UUID] = []
@State
private var paywallReason: PaywallReason?
var body: some View {
debugChanges()
return List {
NameSection(
name: $profileEditor.profile.name,
placeholder: Strings.Placeholders.Profile.name
)
modulesSection
StorageSection(
profileEditor: profileEditor,
paywallReason: $paywallReason
)
UUIDSection(uuid: profileEditor.profile.id)
}
.modifier(PaywallModifier(reason: $paywallReason))
.toolbar(content: toolbarContent)
.navigationTitle(Strings.Global.profile)
.navigationBarBackButtonHidden(true)
.navigationDestination(for: NavigationRoute.self, destination: pushDestination)
}
}
// MARK: -
private extension ProfileEditView {
@ToolbarContentBuilder
func toolbarContent() -> some ToolbarContent {
ToolbarItem(placement: .confirmationAction) {
ProfileSaveButton(
title: Strings.Global.save,
errorModuleIds: $malformedModuleIds
) {
try await flow?.onCommitEditing()
}
}
ToolbarItem(placement: .cancellationAction) {
Button(Strings.Global.cancel, role: .cancel) {
flow?.onCancelEditing()
}
}
}
var modulesSection: some View {
Group {
ForEach(profileEditor.modules, id: \.id, content: moduleRow)
.onMove(perform: moveModules)
.onDelete(perform: removeModules)
addModuleButton
}
.themeSection(
header: Strings.Global.modules,
footer: Strings.Views.Profile.ModuleList.Section.footer
)
}
func moduleRow(for module: any ModuleBuilder) -> some View {
EditorModuleToggle(profileEditor: profileEditor, module: module) {
Button {
push(.moduleDetail(moduleId: module.id))
} label: {
HStack {
Text(module.description(inEditor: profileEditor))
.themeError(malformedModuleIds.contains(module.id))
Spacer()
}
.contentShape(.rect)
}
}
}
var addModuleButton: some View {
let moduleTypes = profileEditor.availableModuleTypes.sorted {
$0.localizedDescription.lowercased() < $1.localizedDescription.lowercased()
}
return Menu {
ForEach(moduleTypes) { selectedType in
Button(selectedType.localizedDescription) {
flow?.onNewModule(selectedType)
}
}
} label: {
Text(Strings.Views.Profile.Rows.addModule)
// .frame(maxWidth: .infinity, alignment: .leading)
}
.disabled(moduleTypes.isEmpty)
}
}
private extension ProfileEditView {
func moveModules(from offsets: IndexSet, to newOffset: Int) {
profileEditor.moveModules(from: offsets, to: newOffset)
}
func removeModules(at offsets: IndexSet) {
profileEditor.removeModules(at: offsets)
}
}
// MARK: - Destinations
private extension ProfileEditView {
enum NavigationRoute: Hashable {
case moduleDetail(moduleId: UUID)
}
@ViewBuilder
func pushDestination(for item: NavigationRoute) -> some View {
switch item {
case .moduleDetail(let moduleId):
ModuleDetailView(
profileEditor: profileEditor,
moduleId: moduleId,
moduleViewFactory: moduleViewFactory
)
.environment(\.navigationPath, $path)
}
}
func push(_ item: NavigationRoute) {
path.append(item)
}
}
#Preview {
NavigationStack {
ProfileEditView(
profileEditor: ProfileEditor(profile: .newMockProfile()),
moduleViewFactory: DefaultModuleViewFactory(registry: Registry()),
path: .constant(NavigationPath())
)
}
.withMockEnvironment()
}
#endif