212 lines
5.9 KiB
Swift
212 lines
5.9 KiB
Swift
//
|
|
// Theme.swift
|
|
// Passepartout-iOS
|
|
//
|
|
// Created by Davide De Rosa on 6/14/18.
|
|
// Copyright (c) 2020 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 UIKit
|
|
import MessageUI
|
|
import StoreKit
|
|
import PassepartoutCore
|
|
|
|
struct Theme {
|
|
struct Palette {
|
|
var primaryBackground = UIColor(rgb: 0x515d71, alpha: 1.0)
|
|
|
|
var accent1 = UIColor(rgb: 0xd69c68, alpha: 1.0)
|
|
|
|
var primaryText: UIColor = {
|
|
guard #available(iOS 13, *) else {
|
|
return .darkText
|
|
}
|
|
return .label
|
|
}()
|
|
|
|
var primaryLightText: UIColor = .white
|
|
|
|
var secondaryText: UIColor = {
|
|
guard #available(iOS 13, *) else {
|
|
return .gray
|
|
}
|
|
return .secondaryLabel
|
|
}()
|
|
|
|
var on: UIColor {
|
|
return accent1
|
|
}
|
|
|
|
var indeterminate: UIColor {
|
|
return secondaryText
|
|
}
|
|
|
|
var off: UIColor {
|
|
return secondaryText
|
|
}
|
|
|
|
// var action = UIColor(red: 214.0 / 255.0, green: 156.0 / 255.0, blue: 104.0 / 255.0, alpha: 1.0)
|
|
var action: UIColor {
|
|
return accent1
|
|
}
|
|
|
|
var accessory: UIColor {
|
|
return accent1.withAlphaComponent(0.7)
|
|
}
|
|
|
|
var destructive = UIColor(red: 0.8, green: 0.27, blue: 0.2, alpha: 1.0)
|
|
}
|
|
|
|
static let current = Theme()
|
|
|
|
var palette: Palette
|
|
|
|
var modalPresentationStyle: UIModalPresentationStyle
|
|
|
|
private init() {
|
|
palette = Palette()
|
|
modalPresentationStyle = .formSheet
|
|
}
|
|
}
|
|
|
|
extension Theme {
|
|
func applyAppearance() {
|
|
let bar = UINavigationBar.appearance()
|
|
bar.barTintColor = palette.primaryBackground
|
|
bar.tintColor = palette.primaryLightText
|
|
bar.titleTextAttributes = [NSAttributedString.Key.foregroundColor: palette.primaryLightText]
|
|
bar.largeTitleTextAttributes = bar.titleTextAttributes
|
|
|
|
let toolbar = UIToolbar.appearance()
|
|
toolbar.barTintColor = palette.primaryBackground
|
|
toolbar.tintColor = palette.primaryLightText
|
|
|
|
let toggle = UISwitch.appearance()
|
|
toggle.onTintColor = palette.accessory
|
|
|
|
let activity = UIActivityIndicatorView.appearance()
|
|
activity.color = palette.accessory
|
|
}
|
|
}
|
|
|
|
extension UIViewController {
|
|
func applyModalPresentation(_ theme: Theme) {
|
|
modalPresentationStyle = theme.modalPresentationStyle
|
|
}
|
|
}
|
|
|
|
//extension UIView {
|
|
// func applyPrimaryBackground(_ theme: Theme) {
|
|
// backgroundColor = theme.palette.primaryBackground
|
|
// }
|
|
//}
|
|
|
|
extension UILabel {
|
|
func apply(_ theme: Theme) {
|
|
font = .preferredFont(forTextStyle: .body)
|
|
textColor = theme.palette.primaryText
|
|
}
|
|
|
|
func applyLight(_ theme: Theme) {
|
|
font = .preferredFont(forTextStyle: .body)
|
|
textColor = theme.palette.primaryLightText
|
|
}
|
|
|
|
func applyAccent(_ theme: Theme) {
|
|
font = .preferredFont(forTextStyle: .body)
|
|
textColor = theme.palette.accent1
|
|
}
|
|
|
|
func applySecondarySize(_ theme: Theme) {
|
|
font = .preferredFont(forTextStyle: .callout)
|
|
}
|
|
}
|
|
|
|
extension UITextField {
|
|
func applyProfileId(_ theme: Theme) {
|
|
placeholder = L10n.Core.Global.Host.TitleInput.placeholder
|
|
clearButtonMode = .always
|
|
keyboardType = .asciiCapable
|
|
returnKeyType = .done
|
|
autocapitalizationType = .none
|
|
autocorrectionType = .no
|
|
}
|
|
}
|
|
|
|
extension UIActivityIndicatorView {
|
|
func applyAccent(_ theme: Theme) {
|
|
color = theme.palette.accent1
|
|
}
|
|
}
|
|
|
|
extension UIBarButtonItem {
|
|
func apply(_ theme: Theme) {
|
|
tintColor = nil
|
|
}
|
|
|
|
func applyAccent(_ theme: Theme) {
|
|
tintColor = theme.palette.accent1
|
|
}
|
|
}
|
|
|
|
extension UIContextualAction {
|
|
func applyNormal(_ theme: Theme) {
|
|
backgroundColor = theme.palette.primaryBackground
|
|
}
|
|
}
|
|
|
|
// XXX: status bar is broken
|
|
extension MFMailComposeViewController {
|
|
func apply(_ theme: Theme) {
|
|
modalPresentationStyle = theme.modalPresentationStyle
|
|
|
|
let bar = navigationBar
|
|
bar.barTintColor = theme.palette.primaryBackground
|
|
bar.tintColor = theme.palette.primaryLightText
|
|
bar.titleTextAttributes = [NSAttributedString.Key.foregroundColor: theme.palette.primaryLightText]
|
|
bar.largeTitleTextAttributes = bar.titleTextAttributes
|
|
}
|
|
}
|
|
|
|
extension Infrastructure.Metadata {
|
|
var logo: UIImage? {
|
|
let bundle = Bundle(for: AppDelegate.self)
|
|
guard let image = AssetImageTypeAlias(named: name, in: bundle, compatibleWith: nil) else {
|
|
return Asset.Providers.placeholder.image
|
|
}
|
|
return image
|
|
}
|
|
|
|
var guidanceString: String? {
|
|
let key = "account.sections.guidance.footer.infrastructure.\(name)"
|
|
let format = NSLocalizedString(key, tableName: "Core", bundle: Bundle(for: AppDelegate.self), comment: "")
|
|
guard !format.isEmpty else {
|
|
return nil
|
|
}
|
|
return String(format: format, locale: .current, description)
|
|
}
|
|
}
|
|
|
|
extension PoolGroup {
|
|
var logo: UIImage? {
|
|
return ImageAsset(name: country.lowercased()).image
|
|
}
|
|
}
|