mirror of
https://github.com/Dimillian/IceCubesApp.git
synced 2025-01-14 15:35:50 +01:00
baf853f46e
* Allow forced translation with DeepL Translation with DeepL can now be forced either per post or on the system level. Signed-off-by: Paul Schuetz <pa.schuetz@web.de> * Require the use of a private API key A private API key of the user is now required to allow "always translate via DeepL". Signed-off-by: Paul Schuetz <pa.schuetz@web.de> * Persist a stored API key An API key is stored even if useOnlyDeepL is disabled. If the API key is empty, the setting is still disabled. Signed-off-by: Paul Schuetz <pa.schuetz@web.de> * Localize the texts Signed-off-by: Paul Schuetz <pa.schuetz@web.de> * Save API key while writing The API key is now saved, even if the app is closed before leaving the translation settings view. Signed-off-by: Paul Schuetz <pa.schuetz@web.de> * Fix build * Fix theme * Transition to KeychainSwift, clean up KeychainHelper is replaced with the already-used KeychainSwift package, the functions are cleaned up so that the process is easier to understand. The deactivateToggleIfNoKey function doesn't change the behavior of the buttons or context menus in the timeline, only demonstrates the necessity of an API key to the user. Consequently, it's only called when the settings view is shown. Signed-off-by: Paul Schuetz <pa.schuetz@web.de> * Swiftformat + fixes --------- Signed-off-by: Paul Schuetz <pa.schuetz@web.de> Co-authored-by: Thomas Ricouard <ricouard77@gmail.com>
307 lines
9.7 KiB
Swift
307 lines
9.7 KiB
Swift
import Account
|
|
import AppAccount
|
|
import DesignSystem
|
|
import Env
|
|
import Foundation
|
|
import Models
|
|
import Network
|
|
import Nuke
|
|
import SwiftUI
|
|
import Timeline
|
|
|
|
struct SettingsTabs: View {
|
|
@Environment(\.dismiss) private var dismiss
|
|
|
|
@EnvironmentObject private var pushNotifications: PushNotificationsService
|
|
@EnvironmentObject private var preferences: UserPreferences
|
|
@EnvironmentObject private var client: Client
|
|
@EnvironmentObject private var currentInstance: CurrentInstance
|
|
@EnvironmentObject private var appAccountsManager: AppAccountsManager
|
|
@EnvironmentObject private var theme: Theme
|
|
|
|
@StateObject private var routerPath = RouterPath()
|
|
|
|
@State private var addAccountSheetPresented = false
|
|
@State private var isEditingAccount = false
|
|
@State private var cachedRemoved = false
|
|
|
|
@Binding var popToRootTab: Tab
|
|
|
|
var body: some View {
|
|
NavigationStack(path: $routerPath.path) {
|
|
Form {
|
|
appSection
|
|
accountsSection
|
|
generalSection
|
|
otherSections
|
|
cacheSection
|
|
}
|
|
.scrollContentBackground(.hidden)
|
|
.background(theme.secondaryBackgroundColor)
|
|
.navigationTitle(Text("settings.title"))
|
|
.navigationBarTitleDisplayMode(.inline)
|
|
.toolbarBackground(theme.primaryBackgroundColor.opacity(0.50), for: .navigationBar)
|
|
.toolbar {
|
|
if UIDevice.current.userInterfaceIdiom == .phone {
|
|
ToolbarItem {
|
|
Button {
|
|
dismiss()
|
|
} label: {
|
|
Text("action.done").bold()
|
|
}
|
|
}
|
|
}
|
|
if UIDevice.current.userInterfaceIdiom == .pad && !preferences.showiPadSecondaryColumn {
|
|
SecondaryColumnToolbarItem()
|
|
}
|
|
}
|
|
.withAppRouter()
|
|
.withSheetDestinations(sheetDestinations: $routerPath.presentedSheet)
|
|
}
|
|
.onAppear {
|
|
routerPath.client = client
|
|
}
|
|
.task {
|
|
if appAccountsManager.currentAccount.oauthToken != nil {
|
|
await currentInstance.fetchCurrentInstance()
|
|
}
|
|
}
|
|
.withSafariRouter()
|
|
.environmentObject(routerPath)
|
|
.onChange(of: $popToRootTab.wrappedValue) { popToRootTab in
|
|
if popToRootTab == .notifications {
|
|
routerPath.path = []
|
|
}
|
|
}
|
|
}
|
|
|
|
private var accountsSection: some View {
|
|
Section("settings.section.accounts") {
|
|
ForEach(appAccountsManager.availableAccounts) { account in
|
|
HStack {
|
|
if isEditingAccount {
|
|
Button {
|
|
Task {
|
|
await logoutAccount(account: account)
|
|
}
|
|
} label: {
|
|
Image(systemName: "trash")
|
|
.renderingMode(.template)
|
|
.tint(.red)
|
|
}
|
|
}
|
|
AppAccountView(viewModel: .init(appAccount: account))
|
|
}
|
|
}
|
|
.onDelete { indexSet in
|
|
if let index = indexSet.first {
|
|
let account = appAccountsManager.availableAccounts[index]
|
|
Task {
|
|
await logoutAccount(account: account)
|
|
}
|
|
}
|
|
}
|
|
if !appAccountsManager.availableAccounts.isEmpty {
|
|
editAccountButton
|
|
}
|
|
addAccountButton
|
|
}
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
}
|
|
|
|
private func logoutAccount(account: AppAccount) async {
|
|
if let token = account.oauthToken,
|
|
let sub = pushNotifications.subscriptions.first(where: { $0.account.token == token })
|
|
{
|
|
let client = Client(server: account.server, oauthToken: token)
|
|
await TimelineCache.shared.clearCache(for: client.id)
|
|
await sub.deleteSubscription()
|
|
appAccountsManager.delete(account: account)
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
private var generalSection: some View {
|
|
Section("settings.section.general") {
|
|
if let instanceData = currentInstance.instance {
|
|
NavigationLink(destination: InstanceInfoView(instance: instanceData)) {
|
|
Label("settings.general.instance", systemImage: "server.rack")
|
|
}
|
|
}
|
|
NavigationLink(destination: DisplaySettingsView()) {
|
|
Label("settings.general.display", systemImage: "paintpalette")
|
|
}
|
|
if HapticManager.shared.supportsHaptics {
|
|
NavigationLink(destination: HapticSettingsView()) {
|
|
Label("settings.general.haptic", systemImage: "waveform.path")
|
|
}
|
|
}
|
|
NavigationLink(destination: remoteLocalTimelinesView) {
|
|
Label("settings.general.remote-timelines", systemImage: "dot.radiowaves.right")
|
|
}
|
|
NavigationLink(destination: ContentSettingsView()) {
|
|
Label("settings.general.content", systemImage: "rectangle.stack")
|
|
}
|
|
NavigationLink(destination: SwipeActionsSettingsView()) {
|
|
Label("settings.general.swipeactions", systemImage: "hand.draw")
|
|
}
|
|
NavigationLink(destination: TranslationSettingsView()) {
|
|
Label("settings.general.translate", systemImage: "captions.bubble")
|
|
}
|
|
Link(destination: URL(string: UIApplication.openSettingsURLString)!) {
|
|
Label("settings.system", systemImage: "gear")
|
|
}
|
|
.tint(theme.labelColor)
|
|
}
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
}
|
|
|
|
private var otherSections: some View {
|
|
Section("settings.section.other") {
|
|
if !ProcessInfo.processInfo.isiOSAppOnMac {
|
|
Picker(selection: $preferences.preferredBrowser) {
|
|
ForEach(PreferredBrowser.allCases, id: \.rawValue) { browser in
|
|
switch browser {
|
|
case .inAppSafari:
|
|
Text("settings.general.browser.in-app").tag(browser)
|
|
case .safari:
|
|
Text("settings.general.browser.system").tag(browser)
|
|
}
|
|
}
|
|
} label: {
|
|
Label("settings.general.browser", systemImage: "network")
|
|
}
|
|
Toggle(isOn: $preferences.inAppBrowserReaderView) {
|
|
Label("settings.general.browser.in-app.readerview", systemImage: "doc.plaintext")
|
|
}
|
|
.disabled(preferences.preferredBrowser != PreferredBrowser.inAppSafari)
|
|
}
|
|
Toggle(isOn: $preferences.isOpenAIEnabled) {
|
|
Label("settings.other.hide-openai", systemImage: "faxmachine")
|
|
}
|
|
Toggle(isOn: $preferences.isSocialKeyboardEnabled) {
|
|
Label("settings.other.social-keyboard", systemImage: "keyboard")
|
|
}
|
|
Toggle(isOn: $preferences.soundEffectEnabled) {
|
|
Label("settings.other.sound-effect", systemImage: "hifispeaker")
|
|
}
|
|
}
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
}
|
|
|
|
private var appSection: some View {
|
|
Section {
|
|
if !ProcessInfo.processInfo.isiOSAppOnMac {
|
|
NavigationLink(destination: IconSelectorView()) {
|
|
Label {
|
|
Text("settings.app.icon")
|
|
} icon: {
|
|
let icon = IconSelectorView.Icon(string: UIApplication.shared.alternateIconName ?? "AppIcon")
|
|
Image(uiImage: .init(named: icon.iconName)!)
|
|
.resizable()
|
|
.frame(width: 25, height: 25)
|
|
.cornerRadius(4)
|
|
}
|
|
}
|
|
}
|
|
|
|
Link(destination: URL(string: "https://github.com/Dimillian/IceCubesApp")!) {
|
|
Label("settings.app.source", systemImage: "link")
|
|
}
|
|
.tint(theme.labelColor)
|
|
|
|
NavigationLink(destination: SupportAppView()) {
|
|
Label("settings.app.support", systemImage: "wand.and.stars")
|
|
}
|
|
|
|
if let reviewURL = URL(string: "https://apps.apple.com/app/id\(AppInfo.appStoreAppId)?action=write-review") {
|
|
Link(destination: reviewURL) {
|
|
Label("settings.rate", systemImage: "link")
|
|
}
|
|
.tint(theme.labelColor)
|
|
}
|
|
|
|
NavigationLink(destination: AboutView()) {
|
|
Label("settings.app.about", systemImage: "info.circle")
|
|
}
|
|
|
|
} header: {
|
|
Text("settings.section.app")
|
|
} footer: {
|
|
if let appVersion = Bundle.main.infoDictionary?["CFBundleShortVersionString"] as? String {
|
|
Text("settings.section.app.footer \(appVersion)").frame(maxWidth: .infinity, alignment: .center)
|
|
}
|
|
}
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
}
|
|
|
|
private var addAccountButton: some View {
|
|
Button {
|
|
addAccountSheetPresented.toggle()
|
|
} label: {
|
|
Text("settings.account.add")
|
|
}
|
|
.sheet(isPresented: $addAccountSheetPresented) {
|
|
AddAccountView()
|
|
}
|
|
}
|
|
|
|
private var editAccountButton: some View {
|
|
Button(role: isEditingAccount ? .none : .destructive) {
|
|
withAnimation {
|
|
isEditingAccount.toggle()
|
|
}
|
|
} label: {
|
|
if isEditingAccount {
|
|
Text("action.done")
|
|
} else {
|
|
Text("account.action.logout")
|
|
}
|
|
}
|
|
}
|
|
|
|
private var remoteLocalTimelinesView: some View {
|
|
Form {
|
|
ForEach(preferences.remoteLocalTimelines, id: \.self) { server in
|
|
Text(server)
|
|
}.onDelete { indexes in
|
|
if let index = indexes.first {
|
|
_ = preferences.remoteLocalTimelines.remove(at: index)
|
|
}
|
|
}
|
|
.onMove(perform: moveTimelineItems)
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
Button {
|
|
routerPath.presentedSheet = .addRemoteLocalTimeline
|
|
} label: {
|
|
Label("settings.timeline.add", systemImage: "badge.plus.radiowaves.right")
|
|
}
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
}
|
|
.navigationTitle("settings.general.remote-timelines")
|
|
.scrollContentBackground(.hidden)
|
|
.background(theme.secondaryBackgroundColor)
|
|
}
|
|
|
|
private func moveTimelineItems(from source: IndexSet, to destination: Int) {
|
|
preferences.remoteLocalTimelines.move(fromOffsets: source, toOffset: destination)
|
|
}
|
|
|
|
private var cacheSection: some View {
|
|
Section("settings.section.cache") {
|
|
if cachedRemoved {
|
|
Text("action.done")
|
|
.transition(.move(edge: .leading))
|
|
} else {
|
|
Button("settings.cache-media.clear", role: .destructive) {
|
|
ImagePipeline.shared.cache.removeAll()
|
|
withAnimation {
|
|
cachedRemoved = true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
}
|
|
}
|