passepartout-apple/Passepartout/Library/Sources/AppUIMain/Views/Profile/iOS/ProfileEditView+iOS.swift
Davide 89d7af4df7
Rethink eligibility checks (#889)
- Allow unrestricted save, but show PurchaseRequiredButton
- Warn however about paid features (FIXME)
- Redesign features in paywall
- Strip already eligible features from paywall
- List required features in restricted alert
- Localize feature descriptions
- Review propagation of paywall modifiers/reasons

Extra:

- Move more domain entities from UILibrary to CommonLibrary
- Default on-demand policy to .any (free feature)
- Fix modals not reappearing after closing with gesture
- Extend UILibrary start-up assertions
2024-11-18 17:43:01 +01:00

195 lines
5.6 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
@Binding
var paywallReason: PaywallReason?
var flow: ProfileCoordinator.Flow?
@State
private var errorModuleIds: Set<UUID> = []
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)
}
.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: $errorModuleIds
) {
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))
if errorModuleIds.contains(module.id) {
ThemeImage(.warning)
} else if profileEditor.isActiveModule(withId: module.id) {
PurchaseRequiredButton(
for: module as? AppFeatureRequiring,
paywallReason: $paywallReason
)
}
Spacer()
}
.contentShape(.rect)
}
.buttonStyle(.plain)
}
}
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)
}
.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()),
paywallReason: .constant(nil)
)
}
.withMockEnvironment()
}
#endif