2023-03-13 13:38:28 +01:00
|
|
|
import Combine
|
2023-01-06 17:14:34 +01:00
|
|
|
import DesignSystem
|
2023-01-22 06:38:30 +01:00
|
|
|
import Env
|
2023-01-17 11:36:01 +01:00
|
|
|
import Models
|
2023-02-18 07:26:48 +01:00
|
|
|
import Network
|
2023-01-06 17:14:34 +01:00
|
|
|
import Status
|
2023-01-17 11:36:01 +01:00
|
|
|
import SwiftUI
|
2023-03-08 19:02:00 +01:00
|
|
|
|
2023-03-31 15:34:24 +02:00
|
|
|
class DisplaySettingsLocalValues: ObservableObject {
|
2023-03-08 19:02:00 +01:00
|
|
|
@Published var tintColor = Theme.shared.tintColor
|
|
|
|
@Published var primaryBackgroundColor = Theme.shared.primaryBackgroundColor
|
|
|
|
@Published var secondaryBackgroundColor = Theme.shared.secondaryBackgroundColor
|
|
|
|
@Published var labelColor = Theme.shared.labelColor
|
2023-03-31 15:34:24 +02:00
|
|
|
@Published var lineSpacing = Theme.shared.lineSpacing
|
|
|
|
@Published var fontSizeScale = Theme.shared.fontSizeScale
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-08 19:02:00 +01:00
|
|
|
private var subscriptions = Set<AnyCancellable>()
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-08 19:02:00 +01:00
|
|
|
init() {
|
|
|
|
$tintColor
|
|
|
|
.debounce(for: .seconds(0.5), scheduler: DispatchQueue.main)
|
2023-03-13 13:38:28 +01:00
|
|
|
.sink(receiveValue: { newColor in Theme.shared.tintColor = newColor })
|
2023-03-08 19:02:00 +01:00
|
|
|
.store(in: &subscriptions)
|
|
|
|
$primaryBackgroundColor
|
|
|
|
.debounce(for: .seconds(0.5), scheduler: DispatchQueue.main)
|
2023-03-13 13:38:28 +01:00
|
|
|
.sink(receiveValue: { newColor in Theme.shared.primaryBackgroundColor = newColor })
|
2023-03-08 19:02:00 +01:00
|
|
|
.store(in: &subscriptions)
|
|
|
|
$secondaryBackgroundColor
|
|
|
|
.debounce(for: .seconds(0.5), scheduler: DispatchQueue.main)
|
2023-03-13 13:38:28 +01:00
|
|
|
.sink(receiveValue: { newColor in Theme.shared.secondaryBackgroundColor = newColor })
|
2023-03-08 19:02:00 +01:00
|
|
|
.store(in: &subscriptions)
|
|
|
|
$labelColor
|
|
|
|
.debounce(for: .seconds(0.5), scheduler: DispatchQueue.main)
|
2023-03-13 13:38:28 +01:00
|
|
|
.sink(receiveValue: { newColor in Theme.shared.labelColor = newColor })
|
2023-03-08 19:02:00 +01:00
|
|
|
.store(in: &subscriptions)
|
2023-03-31 15:34:24 +02:00
|
|
|
$lineSpacing
|
|
|
|
.debounce(for: .seconds(0.5), scheduler: DispatchQueue.main)
|
|
|
|
.sink(receiveValue: { newSpacing in Theme.shared.lineSpacing = newSpacing })
|
|
|
|
.store(in: &subscriptions)
|
|
|
|
$fontSizeScale
|
|
|
|
.debounce(for: .seconds(0.5), scheduler: DispatchQueue.main)
|
|
|
|
.sink(receiveValue: { newScale in Theme.shared.fontSizeScale = newScale })
|
|
|
|
.store(in: &subscriptions)
|
2023-03-08 19:02:00 +01:00
|
|
|
}
|
|
|
|
}
|
2023-01-06 17:14:34 +01:00
|
|
|
|
|
|
|
struct DisplaySettingsView: View {
|
2023-01-30 07:27:06 +01:00
|
|
|
typealias FontState = Theme.FontState
|
2023-02-09 14:22:34 +09:00
|
|
|
|
2023-01-22 16:59:56 +01:00
|
|
|
@Environment(\.colorScheme) private var colorScheme
|
2023-01-06 17:14:34 +01:00
|
|
|
@EnvironmentObject private var theme: Theme
|
2023-01-17 21:08:05 +01:00
|
|
|
@EnvironmentObject private var userPreferences: UserPreferences
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-31 15:34:24 +02:00
|
|
|
@StateObject private var localValues = DisplaySettingsLocalValues()
|
2023-02-09 14:22:34 +09:00
|
|
|
|
2023-01-30 07:27:06 +01:00
|
|
|
@State private var isFontSelectorPresented = false
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-08 19:02:00 +01:00
|
|
|
private let previewStatusViewModel = StatusRowViewModel(status: Status.placeholder(forSettings: true, language: "la"),
|
2023-02-15 08:46:14 +01:00
|
|
|
client: Client(server: ""),
|
|
|
|
routerPath: RouterPath()) // translate from latin button
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-01-06 17:14:34 +01:00
|
|
|
var body: some View {
|
2023-03-26 19:19:59 +02:00
|
|
|
ZStack(alignment: .top) {
|
|
|
|
Form {
|
|
|
|
StatusRowView(viewModel: { previewStatusViewModel })
|
|
|
|
.allowsHitTesting(false)
|
|
|
|
.opacity(0)
|
|
|
|
.hidden()
|
|
|
|
themeSection
|
|
|
|
fontSection
|
|
|
|
layoutSection
|
|
|
|
platformsSection
|
|
|
|
resetSection
|
|
|
|
}
|
|
|
|
.navigationTitle("settings.display.navigation-title")
|
|
|
|
.scrollContentBackground(.hidden)
|
|
|
|
.background(theme.secondaryBackgroundColor)
|
|
|
|
examplePost
|
2023-03-03 08:29:45 +01:00
|
|
|
}
|
|
|
|
}
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-26 19:19:59 +02:00
|
|
|
private var examplePost: some View {
|
|
|
|
VStack(spacing: 0) {
|
2023-03-03 08:29:45 +01:00
|
|
|
StatusRowView(viewModel: { previewStatusViewModel })
|
|
|
|
.allowsHitTesting(false)
|
2023-03-26 19:19:59 +02:00
|
|
|
.padding(.layoutPadding)
|
|
|
|
.background(theme.primaryBackgroundColor)
|
|
|
|
.cornerRadius(8)
|
|
|
|
.padding(.horizontal, .layoutPadding)
|
|
|
|
.padding(.top, .layoutPadding)
|
|
|
|
.background(theme.secondaryBackgroundColor)
|
|
|
|
Rectangle()
|
|
|
|
.fill(theme.secondaryBackgroundColor)
|
|
|
|
.frame(height: 30)
|
|
|
|
.mask(LinearGradient(gradient: Gradient(colors: [theme.secondaryBackgroundColor, .clear]),
|
|
|
|
startPoint: .top, endPoint: .bottom))
|
2023-03-03 08:29:45 +01:00
|
|
|
}
|
|
|
|
}
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
private var themeSection: some View {
|
|
|
|
Section {
|
|
|
|
Toggle("settings.display.theme.systemColor", isOn: $theme.followSystemColorScheme)
|
|
|
|
themeSelectorButton
|
|
|
|
Group {
|
2023-03-31 15:34:24 +02:00
|
|
|
ColorPicker("settings.display.theme.tint", selection: $localValues.tintColor)
|
|
|
|
ColorPicker("settings.display.theme.background", selection: $localValues.primaryBackgroundColor)
|
|
|
|
ColorPicker("settings.display.theme.secondary-background", selection: $localValues.secondaryBackgroundColor)
|
|
|
|
ColorPicker("settings.display.theme.text-color", selection: $localValues.labelColor)
|
2023-02-08 18:09:24 +00:00
|
|
|
}
|
2023-03-03 08:29:45 +01:00
|
|
|
.disabled(theme.followSystemColorScheme)
|
|
|
|
.opacity(theme.followSystemColorScheme ? 0.5 : 1.0)
|
|
|
|
.onChange(of: theme.selectedSet) { _ in
|
2023-03-31 15:34:24 +02:00
|
|
|
localValues.tintColor = theme.tintColor
|
|
|
|
localValues.primaryBackgroundColor = theme.primaryBackgroundColor
|
|
|
|
localValues.secondaryBackgroundColor = theme.secondaryBackgroundColor
|
|
|
|
localValues.labelColor = theme.labelColor
|
2023-03-03 08:29:45 +01:00
|
|
|
}
|
|
|
|
} header: {
|
|
|
|
Text("settings.display.section.theme")
|
|
|
|
} footer: {
|
|
|
|
if theme.followSystemColorScheme {
|
|
|
|
Text("settings.display.section.theme.footer")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
}
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
private var fontSection: some View {
|
|
|
|
Section("settings.display.section.font") {
|
|
|
|
Picker("settings.display.font", selection: .init(get: { () -> FontState in
|
|
|
|
if theme.chosenFont?.fontName == "OpenDyslexic-Regular" {
|
|
|
|
return FontState.openDyslexic
|
|
|
|
} else if theme.chosenFont?.fontName == "AtkinsonHyperlegible-Regular" {
|
|
|
|
return FontState.hyperLegible
|
|
|
|
} else if theme.chosenFont?.fontName == ".AppleSystemUIFontRounded-Regular" {
|
|
|
|
return FontState.SFRounded
|
|
|
|
}
|
|
|
|
return theme.chosenFontData != nil ? FontState.custom : FontState.system
|
|
|
|
}, set: { newValue in
|
|
|
|
switch newValue {
|
|
|
|
case .system:
|
|
|
|
theme.chosenFont = nil
|
|
|
|
case .openDyslexic:
|
|
|
|
theme.chosenFont = UIFont(name: "OpenDyslexic", size: 1)
|
|
|
|
case .hyperLegible:
|
|
|
|
theme.chosenFont = UIFont(name: "Atkinson Hyperlegible", size: 1)
|
|
|
|
case .SFRounded:
|
|
|
|
theme.chosenFont = UIFont.systemFont(ofSize: 1).rounded()
|
|
|
|
case .custom:
|
|
|
|
isFontSelectorPresented = true
|
|
|
|
}
|
|
|
|
})) {
|
|
|
|
ForEach(FontState.allCases, id: \.rawValue) { fontState in
|
|
|
|
Text(fontState.title).tag(fontState)
|
2023-02-15 20:03:31 +01:00
|
|
|
}
|
|
|
|
}
|
2023-03-03 08:29:45 +01:00
|
|
|
.navigationDestination(isPresented: $isFontSelectorPresented, destination: { FontPicker() })
|
2023-02-18 07:26:48 +01:00
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
VStack {
|
2023-03-31 15:34:24 +02:00
|
|
|
Slider(value: $localValues.fontSizeScale, in: 0.5 ... 1.5, step: 0.1)
|
|
|
|
Text("settings.display.font.scaling-\(String(format: "%.1f", localValues.fontSizeScale))")
|
2023-03-03 08:29:45 +01:00
|
|
|
.font(.scaledBody)
|
|
|
|
}
|
|
|
|
.alignmentGuide(.listRowSeparatorLeading) { d in
|
|
|
|
d[.leading]
|
|
|
|
}
|
2023-03-31 15:34:24 +02:00
|
|
|
|
2023-03-26 18:51:15 +02:00
|
|
|
VStack {
|
2023-03-31 15:34:24 +02:00
|
|
|
Slider(value: $localValues.lineSpacing, in: 0.4 ... 10.0, step: 0.2)
|
|
|
|
Text("settings.display.font.line-spacing-\(String(format: "%.1f", localValues.lineSpacing))")
|
2023-03-26 18:51:15 +02:00
|
|
|
.font(.scaledBody)
|
|
|
|
}
|
|
|
|
.alignmentGuide(.listRowSeparatorLeading) { d in
|
|
|
|
d[.leading]
|
|
|
|
}
|
2023-03-03 08:29:45 +01:00
|
|
|
}
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
}
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
private var layoutSection: some View {
|
|
|
|
Section("settings.display.section.display") {
|
|
|
|
Picker("settings.display.avatar.position", selection: $theme.avatarPosition) {
|
|
|
|
ForEach(Theme.AvatarPosition.allCases, id: \.rawValue) { position in
|
|
|
|
Text(position.description).tag(position)
|
2023-01-06 17:14:34 +01:00
|
|
|
}
|
2023-03-03 08:29:45 +01:00
|
|
|
}
|
|
|
|
Picker("settings.display.avatar.shape", selection: $theme.avatarShape) {
|
|
|
|
ForEach(Theme.AvatarShape.allCases, id: \.rawValue) { shape in
|
|
|
|
Text(shape.description).tag(shape)
|
2023-01-07 17:44:25 +01:00
|
|
|
}
|
2023-01-06 17:14:34 +01:00
|
|
|
}
|
2023-03-03 08:29:45 +01:00
|
|
|
Toggle("settings.display.full-username", isOn: $theme.displayFullUsername)
|
|
|
|
Picker("settings.display.status.action-buttons", selection: $theme.statusActionsDisplay) {
|
|
|
|
ForEach(Theme.StatusActionsDisplay.allCases, id: \.rawValue) { buttonStyle in
|
|
|
|
Text(buttonStyle.description).tag(buttonStyle)
|
2023-02-09 14:28:16 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
Picker("settings.display.status.media-style", selection: $theme.statusDisplayStyle) {
|
|
|
|
ForEach(Theme.StatusDisplayStyle.allCases, id: \.rawValue) { buttonStyle in
|
|
|
|
Text(buttonStyle.description).tag(buttonStyle)
|
2023-02-06 18:53:37 +01:00
|
|
|
}
|
|
|
|
}
|
2023-03-03 08:29:45 +01:00
|
|
|
Toggle("settings.display.translate-button", isOn: $userPreferences.showTranslateButton)
|
|
|
|
}
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
}
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
@ViewBuilder
|
|
|
|
private var platformsSection: some View {
|
|
|
|
if UIDevice.current.userInterfaceIdiom == .phone {
|
|
|
|
Section("iPhone") {
|
|
|
|
Toggle("settings.display.show-tab-label", isOn: $userPreferences.showiPhoneTabLabel)
|
|
|
|
}
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
}
|
2023-02-09 14:22:34 +09:00
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
if UIDevice.current.userInterfaceIdiom == .pad {
|
|
|
|
Section("iPad") {
|
|
|
|
Toggle("settings.display.show-ipad-column", isOn: $userPreferences.showiPadSecondaryColumn)
|
2023-01-06 17:14:34 +01:00
|
|
|
}
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
}
|
2023-03-03 08:29:45 +01:00
|
|
|
}
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
private var resetSection: some View {
|
|
|
|
Section {
|
|
|
|
Button {
|
|
|
|
theme.followSystemColorScheme = true
|
|
|
|
theme.selectedSet = colorScheme == .dark ? .iceCubeDark : .iceCubeLight
|
|
|
|
theme.avatarShape = .rounded
|
|
|
|
theme.avatarPosition = .top
|
|
|
|
theme.statusActionsDisplay = .full
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-31 15:34:24 +02:00
|
|
|
localValues.tintColor = theme.tintColor
|
|
|
|
localValues.primaryBackgroundColor = theme.primaryBackgroundColor
|
|
|
|
localValues.secondaryBackgroundColor = theme.secondaryBackgroundColor
|
|
|
|
localValues.labelColor = theme.labelColor
|
2023-03-13 13:38:28 +01:00
|
|
|
|
2023-03-03 08:29:45 +01:00
|
|
|
} label: {
|
|
|
|
Text("settings.display.restore")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
2023-01-06 17:14:34 +01:00
|
|
|
}
|
2023-02-09 14:22:34 +09:00
|
|
|
|
2023-01-06 17:14:34 +01:00
|
|
|
private var themeSelectorButton: some View {
|
|
|
|
NavigationLink(destination: ThemePreviewView()) {
|
|
|
|
HStack {
|
2023-01-19 18:14:08 +01:00
|
|
|
Text("settings.display.section.theme")
|
2023-01-06 17:14:34 +01:00
|
|
|
Spacer()
|
|
|
|
Text(theme.selectedSet.rawValue)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|