mirror of
https://github.com/passepartoutvpn/passepartout-apple.git
synced 2025-01-09 02:02:50 +00:00
80d40c3161
- Show detail side by side rather than navigate - Fix scrolling in purchased view
144 lines
3.4 KiB
Swift
144 lines
3.4 KiB
Swift
//
|
|
// SettingsView.swift
|
|
// Passepartout
|
|
//
|
|
// Created by Davide De Rosa on 11/8/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 CommonUtils
|
|
import PassepartoutKit
|
|
import SwiftUI
|
|
import UILibrary
|
|
|
|
enum Detail {
|
|
case credits
|
|
|
|
case donate
|
|
|
|
case other
|
|
|
|
case purchased
|
|
}
|
|
|
|
struct SettingsView: View {
|
|
let tunnel: ExtendedTunnel
|
|
|
|
@Namespace
|
|
private var masterScope
|
|
|
|
@Namespace
|
|
private var detailScope
|
|
|
|
@FocusState
|
|
private var focus: Detail?
|
|
|
|
@State
|
|
private var detail: Detail?
|
|
|
|
var body: some View {
|
|
HStack {
|
|
masterView
|
|
.frame(maxWidth: .infinity)
|
|
.focused($focus, equals: .other)
|
|
|
|
DetailView(detail: detail)
|
|
.frame(maxWidth: .infinity)
|
|
}
|
|
.onChange(of: focus) {
|
|
guard focus != nil else {
|
|
return
|
|
}
|
|
detail = focus
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension SettingsView {
|
|
var masterView: some View {
|
|
List {
|
|
creditsSection
|
|
diagnosticsSection
|
|
aboutSection
|
|
}
|
|
.themeList()
|
|
}
|
|
|
|
var creditsSection: some View {
|
|
Group {
|
|
Button(Strings.Views.About.Credits.title) {}
|
|
.focused($focus, equals: .credits)
|
|
Button(Strings.Views.Donate.title) {}
|
|
.focused($focus, equals: .donate)
|
|
}
|
|
.themeSection(header: Strings.Unlocalized.appName)
|
|
}
|
|
|
|
var diagnosticsSection: some View {
|
|
Group {
|
|
NavigationLink(Strings.Views.Diagnostics.Rows.app, value: AppCoordinatorRoute.appLog)
|
|
NavigationLink(Strings.Views.Diagnostics.Rows.tunnel, value: AppCoordinatorRoute.tunnelLog)
|
|
LogsPrivateDataToggle()
|
|
}
|
|
.themeSection(header: Strings.Views.Diagnostics.title)
|
|
}
|
|
|
|
var aboutSection: some View {
|
|
Group {
|
|
Button(Strings.Views.Purchased.title) {}
|
|
.focused($focus, equals: .purchased)
|
|
Text(Strings.Global.Nouns.version)
|
|
.themeTrailingValue(BundleConfiguration.mainVersionString)
|
|
}
|
|
.themeSection(header: Strings.Views.About.title)
|
|
}
|
|
}
|
|
|
|
private struct DetailView: View {
|
|
let detail: Detail?
|
|
|
|
var body: some View {
|
|
switch detail {
|
|
case .credits:
|
|
CreditsView()
|
|
.themeList()
|
|
|
|
case .donate:
|
|
DonateView(modifier: DonateViewModifier())
|
|
|
|
case .purchased:
|
|
PurchasedView()
|
|
.themeList()
|
|
|
|
default:
|
|
VStack {}
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: -
|
|
|
|
#Preview {
|
|
SettingsView(tunnel: .mock)
|
|
.themeNavigationStack()
|
|
.withMockEnvironment()
|
|
}
|