// // Theme.swift // Passepartout // // Created by Davide De Rosa on 2/24/22. // Copyright (c) 2022 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 . // import SwiftUI import PassepartoutCore extension View { var themeIdiom: UIUserInterfaceIdiom { UIDevice.current.userInterfaceIdiom } var themeIsiPadPortrait: Bool { #if targetEnvironment(macCatalyst) false #else let device: UIDevice = .current return device.userInterfaceIdiom == .pad && device.orientation.isPortrait #endif } var themeIsiPadMultitasking: Bool { #if targetEnvironment(macCatalyst) false #else guard #available(iOS 15, *) else { return false } return UIDevice.current.userInterfaceIdiom == .pad #endif } } // MARK: Global extension View { func themeGlobal() -> some View { themeNavigationViewStyle() .themeTint() .themeToggleStyle() .menuStyle(.borderlessButton) } @ViewBuilder private func themeTint() -> some View { if #available(iOS 15, *) { tint(.accentColor) } else { self } } @ViewBuilder private func themeToggleStyle() -> some View { if #available(iOS 15, *) { toggleStyle(.switch) } else { toggleStyle(SwitchToggleStyle(tint: .accentColor)) } } @ViewBuilder private func themeNavigationViewStyle() -> some View { switch themeIdiom { case .phone: navigationViewStyle(.stack) default: navigationViewStyle(.automatic) } } func themePrimaryView() -> some View { #if targetEnvironment(macCatalyst) navigationBarTitleDisplayMode(.inline) #else navigationBarTitleDisplayMode(.large) .navigationTitle(Unlocalized.appName) #endif } func themeSecondaryView() -> some View { navigationBarTitleDisplayMode(.inline) .listStyle(.insetGrouped) } } // MARK: Colors extension View { fileprivate var themePrimaryBackgroundColor: Color { Color(Asset.Assets.primaryColor.color) } fileprivate var themeSecondaryColor: Color { .secondary } fileprivate var themeLightTextColor: Color { Color(Asset.Assets.lightTextColor.color) } fileprivate var themeErrorColor: Color { .red } private func themeColor(_ string: String?, validator: (String) throws -> Void) -> Color? { guard let string = string else { return nil } do { try validator(string) return nil } catch { return themeErrorColor } } } // MARK: Images extension View { var themeAssetsLogoImage: String { "logo" } var themeCheckmarkImage: String { "checkmark" } var themeShareImage: String { "square.and.arrow.up" } var themeCloseImage: String { "xmark" } var themeConceilImage: String { "eye.slash" } var themeRevealImage: String { "eye" } // MARK: Organizer func themeAssetsProviderImage(_ providerName: ProviderName) -> String { "providers/\(providerName)" } func themeAssetsCountryImage(_ countryCode: String) -> String { "flags/\(countryCode.lowercased())" } var themeProviderImage: String { "externaldrive.connected.to.line.below" } var themeHostFilesImage: String { "folder" } var themeHostTextImage: String { "text.justify" } var themeInfoMenuImage: String { "info.circle" } var themeDonateImage: String { "giftcard" } var themeRedditImage: String { "person.3" } var themeWriteReviewImage: String { "star" } var themeAddMenuImage: String { "plus" } var themeProfileActiveImage: String { "checkmark.circle" } var themeProfileConnectedImage: String { "circle.fill" } var themeProfileInactiveImage: String { "circle" } // MARK: Profile var themeSettingsMenuImage: String { "ellipsis.circle" } var themeShortcutsImage: String { "mic" } var themeRenameProfileImage: String { "highlighter" // "character.cursor.ibeam" } var themeDuplicateImage: String { "doc.on.doc" } var themeUninstallImage: String { "arrow.uturn.down" } var themeDeleteImage: String { "trash" } var themeVPNProtocolImage: String { "bolt" // "waveform.path.ecg" // "message.and.waveform.fill" // "pc" // "captions.bubble.fill" } var themeEndpointImage: String { "link" } var themeAccountImage: String { "person" } var themeProviderLocationImage: String { "location" } var themeProviderPresetImage: String { "slider.horizontal.3" } var themeProviderRefreshImage: String { "arrow.clockwise" } var themeNetworkSettingsImage: String { // "network" "globe" } var themeOnDemandImage: String { "wifi" } var themeDiagnosticsImage: String { "bandage.fill" } var themeFAQImage: String { "questionmark.diamond" } func themeFavoritesImage(_ active: Bool) -> String { active ? "bookmark.fill" : "bookmark" } func themeFavoriteActionImage(_ doFavorite: Bool) -> String { doFavorite ? "bookmark" : "bookmark.slash.fill" } } extension String { var asAssetImage: Image { Image(self) } var asSystemImage: Image { Image(systemName: self) } } // MARK: Styles extension View { func themeAccentForegroundStyle() -> some View { foregroundColor(.accentColor) } var themePrimaryBackground: some View { themePrimaryBackgroundColor .ignoresSafeArea() } func themeSecondaryTextStyle() -> some View { foregroundColor(themeSecondaryColor) } func themeLightTextStyle() -> some View { foregroundColor(themeLightTextColor) } @available(iOS 15, *) func themePrimaryTintStyle() -> some View { tint(themePrimaryBackgroundColor) } func themeTextButtonStyle() -> some View { accentColor(.primary) } func themeLongTextStyle() -> some View { lineLimit(1) .truncationMode(.middle) } func themeInformativeTextStyle() -> some View { multilineTextAlignment(.center) .font(.title) .foregroundColor(themeSecondaryColor) } func themeDebugLogStyle() -> some View { font(.system(size: 13, weight: .medium, design: .monospaced)) } } // MARK: Animations extension View { func themeAnimation(on value: V) -> some View { animation(.default, value: value) } } extension Binding { func themeAnimation() -> Binding { animation(.default) } } // MARK: Shortcuts extension View { func themeCloseItem(presentationMode: Binding) -> some ToolbarContent { ToolbarItem(placement: .cancellationAction) { Button { presentationMode.wrappedValue.dismiss() } label: { themeCloseImage.asSystemImage } } } func themeCloseItem(isPresented: Binding) -> some ToolbarContent { ToolbarItem(placement: .cancellationAction) { Button { isPresented.wrappedValue = false } label: { themeCloseImage.asSystemImage } } } func themeSaveButtonLabel() -> some View { // themeCheckmarkImage.asSystemImage Text(L10n.Global.Strings.save) } // func themeDoneButtonLabel() -> some View { //// themeCheckmarkImage.asSystemImage // Text(L10n.Global.Strings.ok) // } func themeTextPicker(_ title: String, selection: Binding, values: [T], description: @escaping (T) -> String) -> some View { StyledPicker(title: title, selection: selection, values: values) { Text(description($0)) } selectionLabel: { Text(description($0)) .foregroundColor(themeSecondaryColor) } listStyle: { .insetGrouped } } func themeLongContentLinkDefault(_ title: String, content: Binding) -> some View { LongContentLink(title, content: content) { Text($0) .foregroundColor(themeSecondaryColor) } } func themeLongContentLink(_ title: String, content: Binding, withPreview preview: String? = nil) -> some View { LongContentLink(title, content: content, preview: preview) { Text(preview != nil ? $0 : "") .foregroundColor(themeSecondaryColor) } } @ViewBuilder func themeErrorMessage(_ message: String?) -> some View { if let message = message { if message.last != "." { Text("\(message).") .foregroundColor(themeErrorColor) } else { Text(message) .foregroundColor(themeErrorColor) } } else { EmptyView() } } } // MARK: Validation extension View { func themeValidProfileName() -> some View { autocapitalization(.none) .disableAutocorrection(true) } func themeValidURL(_ urlString: String?) -> some View { themeValidating(urlString, validator: Validators.url) .keyboardType(.asciiCapable) .autocapitalization(.none) .disableAutocorrection(true) } func themeValidIPAddress(_ ipAddress: String?) -> some View { themeValidating(ipAddress, validator: Validators.ipAddress) .keyboardType(.numbersAndPunctuation) .autocapitalization(.none) .disableAutocorrection(true) } func themeValidSocketPort() -> some View { keyboardType(.numberPad) } func themeValidDomainName(_ domainName: String?) -> some View { themeValidating(domainName, validator: Validators.domainName) .keyboardType(.asciiCapable) .autocapitalization(.none) .disableAutocorrection(true) } func themeValidDNSOverTLSServerName(_ string: String?) -> some View { themeValidating(string, validator: Validators.dnsOverTLSServerName) .keyboardType(.asciiCapable) .autocapitalization(.none) .disableAutocorrection(true) } func themeValidSSID(_ text: String?) -> some View { themeValidating(text, validator: Validators.notEmpty) .keyboardType(.asciiCapable) .autocapitalization(.none) .disableAutocorrection(true) } private func themeValidating(_ string: String?, validator: (String) throws -> Void) -> some View { foregroundColor(themeColor(string, validator: validator)) } } // MARK: Hacks extension View { // @available(*, deprecated, message: "mitigates multiline text truncation (1.0 does not work though)") func xxxThemeTruncation() -> some View { minimumScaleFactor(0.5) } }