yattee/Shared/Settings/PlayerSettings.swift

499 lines
18 KiB
Swift
Raw Normal View History

2021-09-25 08:18:22 +00:00
import Defaults
import SwiftUI
2022-01-06 15:02:53 +00:00
struct PlayerSettings: View {
2021-11-03 23:40:01 +00:00
@Default(.instances) private var instances
@Default(.playerInstanceID) private var playerInstanceID
2022-01-06 15:02:53 +00:00
2021-11-03 23:00:17 +00:00
@Default(.playerSidebar) private var playerSidebar
2022-12-19 12:35:37 +00:00
2021-11-03 23:14:09 +00:00
@Default(.showKeywords) private var showKeywords
2024-08-20 18:38:18 +00:00
@Default(.showComments) private var showComments
#if !os(tvOS)
@Default(.showScrollToTopInComments) private var showScrollToTopInComments
@Default(.collapsedLinesDescription) private var collapsedLinesDescription
2023-11-27 22:49:18 +00:00
@Default(.exitFullscreenOnEOF) private var exitFullscreenOnEOF
#endif
2023-04-22 17:22:13 +00:00
@Default(.expandVideoDescription) private var expandVideoDescription
@Default(.pauseOnHidingPlayer) private var pauseOnHidingPlayer
@Default(.closeVideoOnEOF) private var closeVideoOnEOF
#if os(iOS)
@Default(.enterFullscreenInLandscape) private var enterFullscreenInLandscape
@Default(.lockPortraitWhenBrowsing) private var lockPortraitWhenBrowsing
@Default(.rotateToLandscapeOnEnterFullScreen) private var rotateToLandscapeOnEnterFullScreen
#endif
@Default(.closePiPOnNavigation) private var closePiPOnNavigation
@Default(.closePiPOnOpeningPlayer) private var closePiPOnOpeningPlayer
2022-08-26 20:17:21 +00:00
@Default(.closePlayerOnOpeningPiP) private var closePlayerOnOpeningPiP
#if !os(macOS)
@Default(.pauseOnEnteringBackground) private var pauseOnEnteringBackground
@Default(.closePiPAndOpenPlayerOnEnteringForeground) private var closePiPAndOpenPlayerOnEnteringForeground
#endif
2021-11-03 23:00:17 +00:00
2022-03-20 20:31:19 +00:00
@Default(.enableReturnYouTubeDislike) private var enableReturnYouTubeDislike
@Default(.showRelated) private var showRelated
@Default(.showInspector) private var showInspector
@Default(.showChapters) private var showChapters
@Default(.showChapterThumbnails) private var showThumbnails
@Default(.showChapterThumbnailsOnlyWhenDifferent) private var showThumbnailsOnlyWhenDifferent
@Default(.expandChapters) private var expandChapters
@Default(.captionsAutoShow) private var captionsAutoShow
@Default(.captionsDefaultLanguageCode) private var captionsDefaultLanguageCode
2024-05-20 12:40:25 +00:00
@Default(.captionsFallbackLanguageCode) private var captionsFallbackLanguageCode
2024-05-20 13:34:31 +00:00
@Default(.captionsFontScaleSize) private var captionsFontScaleSize
2024-05-20 15:17:52 +00:00
@Default(.captionsFontColor) private var captionsFontColor
2022-12-18 18:39:03 +00:00
@ObservedObject private var accounts = AccountsModel.shared
2022-03-20 20:31:19 +00:00
2021-11-03 23:00:17 +00:00
#if os(iOS)
private var idiom: UIUserInterfaceIdiom {
UIDevice.current.userInterfaceIdiom
}
#endif
2021-09-25 08:18:22 +00:00
2024-05-20 12:20:08 +00:00
#if os(tvOS)
2024-05-20 12:40:25 +00:00
@State private var isShowingDefaultLanguagePicker = false
@State private var isShowingFallbackLanguagePicker = false
2024-05-20 12:20:08 +00:00
#endif
2021-09-25 08:18:22 +00:00
var body: some View {
2021-11-04 22:01:27 +00:00
Group {
2022-01-06 15:02:53 +00:00
#if os(macOS)
sections
2022-01-06 15:02:53 +00:00
Spacer()
2021-11-04 22:01:27 +00:00
#else
2022-01-06 15:02:53 +00:00
List {
sections
2021-11-04 22:01:27 +00:00
}
2022-01-06 15:02:53 +00:00
#endif
}
#if os(tvOS)
.frame(maxWidth: 1000)
#elseif os(iOS)
.listStyle(.insetGrouped)
#endif
.navigationTitle("Player")
}
2021-11-03 23:40:01 +00:00
2022-01-06 15:02:53 +00:00
private var sections: some View {
Group {
2022-09-04 15:28:30 +00:00
Section(header: SettingsHeader(text: "Playback".localized())) {
2022-11-11 19:34:20 +00:00
if !accounts.isEmpty {
sourcePicker
}
2022-01-06 15:02:53 +00:00
pauseOnHidingPlayerToggle
closeVideoOnEOFToggle
#if os(macOS)
2023-11-27 22:49:18 +00:00
exitFullscreenOnEOFToggle
#endif
#if !os(macOS)
pauseOnEnteringBackgroundToogle
#endif
2022-12-19 11:08:27 +00:00
}
#if !os(tvOS)
Section(header: SettingsHeader(text: "Info".localized())) {
expandVideoDescriptionToggle
collapsedLineDescriptionStepper
showRelatedToggle
#if os(macOS)
HStack {
Text("Inspector")
inspectorVisibilityPicker
}
.padding(.leading, 20)
#else
inspectorVisibilityPicker
#endif
}
#endif
Section(header: SettingsHeader(text: "Captions".localized())) {
2024-05-20 13:34:31 +00:00
#if os(tvOS)
Text("Size").font(.subheadline)
#endif
captionsFontScaleSizePicker
2024-05-20 15:17:52 +00:00
#if os(tvOS)
Text("Color").font(.subheadline)
#endif
captionsFontColorPicker
showCaptionsAutoShowToggle
2024-05-20 13:34:31 +00:00
2024-05-20 12:20:08 +00:00
#if !os(tvOS)
captionDefaultLanguagePicker
2024-05-20 12:40:25 +00:00
captionFallbackLanguagePicker
2024-05-20 12:20:08 +00:00
#else
2024-05-20 12:40:25 +00:00
Button(action: { isShowingDefaultLanguagePicker = true }) {
2024-05-20 12:20:08 +00:00
HStack {
Text("Default language")
Spacer()
Text("\(LanguageCodes(rawValue: captionsDefaultLanguageCode)!.description.capitalized) (\(captionsDefaultLanguageCode))").foregroundColor(.secondary)
}
}
2024-05-20 12:40:25 +00:00
.frame(maxWidth: .infinity).sheet(isPresented: $isShowingDefaultLanguagePicker) {
defaultLanguagePickerTVOS(
2024-05-20 12:20:08 +00:00
selectedLanguage: $captionsDefaultLanguageCode,
2024-05-20 12:40:25 +00:00
isShowing: $isShowingDefaultLanguagePicker
)
}
Button(action: { isShowingFallbackLanguagePicker = true }) {
HStack {
Text("Fallback language")
Spacer()
Text("\(LanguageCodes(rawValue: captionsFallbackLanguageCode)!.description.capitalized) (\(captionsFallbackLanguageCode))").foregroundColor(.secondary)
}
}
.frame(maxWidth: .infinity).sheet(isPresented: $isShowingDefaultLanguagePicker) {
fallbackLanguagePickerTVOS(
selectedLanguage: $captionsFallbackLanguageCode,
isShowing: $isShowingFallbackLanguagePicker
2024-05-20 12:20:08 +00:00
)
}
#endif
}
#if !os(tvOS)
Section(header: SettingsHeader(text: "Chapters".localized())) {
showChaptersToggle
showThumbnailsToggle
showThumbnailsWhenDifferentToggle
expandChaptersToggle
}
#endif
let interface = Section(header: SettingsHeader(text: "Interface".localized())) {
2022-01-06 15:02:53 +00:00
#if os(iOS)
if idiom == .pad {
2021-11-04 22:01:27 +00:00
sidebarPicker
}
#endif
2022-01-06 15:02:53 +00:00
#if os(macOS)
sidebarPicker
#endif
2022-11-11 19:34:20 +00:00
if !accounts.isEmpty {
keywordsToggle
2024-08-20 18:38:18 +00:00
commentsToggle
#if !os(tvOS)
showScrollToTopInCommentsToggle
#endif
returnYouTubeDislikeToggle
}
2022-01-06 15:02:53 +00:00
}
#if os(tvOS)
if !accounts.isEmpty {
interface
}
#elseif os(macOS)
interface
#elseif os(iOS)
if idiom == .pad || !accounts.isEmpty {
interface
}
#endif
2022-01-06 15:02:53 +00:00
#if os(iOS)
Section(header: SettingsHeader(text: "Fullscreen".localized())) {
if Constants.isIPad {
2022-01-06 15:02:53 +00:00
enterFullscreenInLandscapeToggle
}
exitFullscreenOnEOFToggle
rotateToLandscapeOnEnterFullScreenPicker
}
2021-11-04 22:01:27 +00:00
#endif
2022-09-04 15:28:30 +00:00
Section(header: SettingsHeader(text: "Picture in Picture".localized())) {
closePiPOnNavigationToggle
closePiPOnOpeningPlayerToggle
2022-08-26 20:17:21 +00:00
closePlayerOnOpeningPiPToggle
#if !os(macOS)
closePiPAndOpenPlayerOnEnteringForegroundToggle
#endif
}
2022-01-06 15:02:53 +00:00
}
2021-11-03 23:14:09 +00:00
}
private var videoDetailsHeaderPadding: Double {
#if os(macOS)
5.0
#else
0.0
#endif
}
2021-11-04 22:01:27 +00:00
private var sourcePicker: some View {
Picker("Source", selection: $playerInstanceID) {
2022-09-04 15:28:30 +00:00
Text("Instance of current account").tag(String?.none)
2021-11-03 23:40:01 +00:00
2021-11-04 22:01:27 +00:00
ForEach(instances) { instance in
2021-12-04 19:35:41 +00:00
Text(instance.description).tag(Optional(instance.id))
2021-11-03 23:40:01 +00:00
}
}
2022-08-06 14:28:05 +00:00
.modifier(SettingsPickerModifier())
}
2021-11-03 23:14:09 +00:00
private var sidebarPicker: some View {
Picker("Sidebar", selection: $playerSidebar) {
#if os(macOS)
2022-01-06 15:02:53 +00:00
Text("Show sidebar").tag(PlayerSidebarSetting.always)
2021-11-03 23:14:09 +00:00
#endif
2021-11-03 23:00:17 +00:00
#if os(iOS)
2021-11-03 23:14:09 +00:00
Text("Show sidebar when space permits").tag(PlayerSidebarSetting.whenFits)
2021-09-25 08:18:22 +00:00
#endif
2021-11-03 23:14:09 +00:00
2022-01-06 15:02:53 +00:00
Text("Hide sidebar").tag(PlayerSidebarSetting.never)
2021-09-25 08:18:22 +00:00
}
2022-08-06 14:28:05 +00:00
.modifier(SettingsPickerModifier())
2021-09-25 08:18:22 +00:00
}
2021-11-04 22:01:27 +00:00
2024-08-20 18:38:18 +00:00
private var commentsToggle: some View {
Toggle("Show comments", isOn: $showComments)
}
#if !os(tvOS)
private var showScrollToTopInCommentsToggle: some View {
2024-08-20 18:38:18 +00:00
Toggle("Show scroll to top button in comments", isOn: $showScrollToTopInComments).disabled(!showComments)
}
#endif
2021-11-04 22:01:27 +00:00
private var keywordsToggle: some View {
2022-01-06 15:02:53 +00:00
Toggle("Show keywords", isOn: $showKeywords)
2021-11-04 22:01:27 +00:00
}
2021-12-17 20:01:05 +00:00
2023-04-22 17:22:13 +00:00
private var expandVideoDescriptionToggle: some View {
Toggle("Open video description expanded", isOn: $expandVideoDescription)
}
#if !os(tvOS)
private var collapsedLineDescriptionStepper: some View {
LazyVStack {
Stepper(value: $collapsedLinesDescription, in: 0 ... 10) {
Text("Description preview")
#if os(macOS)
Spacer()
#endif
if collapsedLinesDescription == 0 {
Text("No preview")
} else {
Text("\(collapsedLinesDescription) lines")
}
2023-11-24 13:09:40 +00:00
}
}
}
#endif
2022-03-20 20:31:19 +00:00
private var returnYouTubeDislikeToggle: some View {
Toggle("Enable Return YouTube Dislike", isOn: $enableReturnYouTubeDislike)
}
private var pauseOnHidingPlayerToggle: some View {
Toggle("Pause when player is closed", isOn: $pauseOnHidingPlayer)
}
private var closeVideoOnEOFToggle: some View {
Toggle("Close video and player on end", isOn: $closeVideoOnEOF)
}
2023-11-27 22:49:18 +00:00
#if !os(tvOS)
private var exitFullscreenOnEOFToggle: some View {
Toggle("Exit fullscreen on end", isOn: $exitFullscreenOnEOF)
.disabled(closeVideoOnEOF)
}
#endif
#if !os(macOS)
private var pauseOnEnteringBackgroundToogle: some View {
Toggle("Pause when entering background", isOn: $pauseOnEnteringBackground)
}
#endif
#if os(iOS)
private var enterFullscreenInLandscapeToggle: some View {
Toggle("Enter fullscreen in landscape orientation", isOn: $enterFullscreenInLandscape)
.disabled(lockPortraitWhenBrowsing)
}
private var rotateToLandscapeOnEnterFullScreenPicker: some View {
Picker("Default orientation", selection: $rotateToLandscapeOnEnterFullScreen) {
Text("Landscape left").tag(FullScreenRotationSetting.landscapeLeft)
Text("Landscape right").tag(FullScreenRotationSetting.landscapeRight)
}
.modifier(SettingsPickerModifier())
}
#endif
private var closePiPOnNavigationToggle: some View {
Toggle("Close PiP when starting playing other video", isOn: $closePiPOnNavigation)
}
private var closePiPOnOpeningPlayerToggle: some View {
Toggle("Close PiP when player is opened", isOn: $closePiPOnOpeningPlayer)
}
2022-08-26 20:17:21 +00:00
private var closePlayerOnOpeningPiPToggle: some View {
Toggle("Close player when starting PiP", isOn: $closePlayerOnOpeningPiP)
}
#if !os(macOS)
private var closePiPAndOpenPlayerOnEnteringForegroundToggle: some View {
Toggle("Close PiP and open player when application enters foreground", isOn: $closePiPAndOpenPlayerOnEnteringForeground)
}
#endif
2024-05-20 12:20:08 +00:00
private var showCaptionsAutoShowToggle: some View {
Toggle("Always show captions", isOn: $captionsAutoShow)
}
2024-05-20 13:34:31 +00:00
private var captionsFontScaleSizePicker: some View {
Picker("Size", selection: $captionsFontScaleSize) {
Text("Small").tag(String("0.725"))
2024-05-20 15:17:52 +00:00
Text("Medium").tag(String("1.0"))
Text("Large").tag(String("1.5"))
2024-05-20 13:34:31 +00:00
}
.onChange(of: captionsFontScaleSize) { _ in
2024-05-20 15:17:52 +00:00
PlayerModel.shared.mpvBackend.client.setSubFontSize(scaleSize: captionsFontScaleSize)
}
#if os(macOS)
.labelsHidden()
#endif
}
private var captionsFontColorPicker: some View {
Picker("Color", selection: $captionsFontColor) {
Text("White").tag(String("#FFFFFF"))
Text("Yellow").tag(String("#FFFF00"))
Text("Red").tag(String("#FF0000"))
Text("Orange").tag(String("#FFA500"))
Text("Green").tag(String("#008000"))
Text("Blue").tag(String("#0000FF"))
}
.onChange(of: captionsFontColor) { _ in
PlayerModel.shared.mpvBackend.client.setSubFontColor(color: captionsFontColor)
2024-05-20 13:34:31 +00:00
}
#if os(macOS)
.labelsHidden()
#endif
}
#if !os(tvOS)
2024-05-20 12:20:08 +00:00
private var captionDefaultLanguagePicker: some View {
Picker("Default language", selection: $captionsDefaultLanguageCode) {
ForEach(LanguageCodes.allCases, id: \.self) { language in
Text("\(language.description.capitalized) (\(language.rawValue))").tag(language.rawValue)
}
}
#if os(macOS)
.labelsHidden()
#endif
}
2024-05-20 12:40:25 +00:00
private var captionFallbackLanguagePicker: some View {
Picker("Fallback language", selection: $captionsFallbackLanguageCode) {
ForEach(LanguageCodes.allCases, id: \.self) { language in
Text("\(language.description.capitalized) (\(language.rawValue))").tag(language.rawValue)
}
}
#if os(macOS)
.labelsHidden()
#endif
}
2024-05-20 12:20:08 +00:00
#else
2024-05-20 12:40:25 +00:00
struct defaultLanguagePickerTVOS: View {
2024-05-20 12:20:08 +00:00
@Binding var selectedLanguage: String
@Binding var isShowing: Bool
var body: some View {
NavigationView {
List(LanguageCodes.allCases, id: \.self) { language in
Button(action: {
selectedLanguage = language.rawValue
isShowing = false
}) {
Text("\(language.description.capitalized) (\(language.rawValue))")
}
}
.navigationTitle("Select Default Language")
}
}
}
2024-05-20 12:40:25 +00:00
struct fallbackLanguagePickerTVOS: View {
@Binding var selectedLanguage: String
@Binding var isShowing: Bool
var body: some View {
NavigationView {
List(LanguageCodes.allCases, id: \.self) { language in
Button(action: {
selectedLanguage = language.rawValue
isShowing = false
}) {
Text("\(language.description.capitalized) (\(language.rawValue))")
}
}
.navigationTitle("Select Fallback Language")
}
}
}
2024-05-20 12:20:08 +00:00
#endif
2024-05-20 12:20:08 +00:00
#if !os(tvOS)
private var inspectorVisibilityPicker: some View {
Picker("Inspector", selection: $showInspector) {
Text("Always").tag(ShowInspectorSetting.always)
Text("Only for local files and URLs").tag(ShowInspectorSetting.onlyLocal)
}
#if os(macOS)
.labelsHidden()
#endif
}
private var showChaptersToggle: some View {
Toggle("Show chapters", isOn: $showChapters)
}
private var showThumbnailsToggle: some View {
Toggle("Show thumbnails", isOn: $showThumbnails)
.disabled(!showChapters)
.foregroundColor(showChapters ? .primary : .secondary)
}
private var showThumbnailsWhenDifferentToggle: some View {
Toggle("Show thumbnails only when unique", isOn: $showThumbnailsOnlyWhenDifferent)
.disabled(!showChapters || !showThumbnails)
.foregroundColor(showChapters && showThumbnails ? .primary : .secondary)
}
private var expandChaptersToggle: some View {
Toggle("Open vertical chapters expanded", isOn: $expandChapters)
.disabled(!showChapters)
.foregroundColor(showChapters ? .primary : .secondary)
}
private var showRelatedToggle: some View {
Toggle("Related", isOn: $showRelated)
}
#endif
2021-11-04 22:01:27 +00:00
}
struct PlayerSettings_Previews: PreviewProvider {
2021-11-04 22:01:27 +00:00
static var previews: some View {
VStack(alignment: .leading) {
PlayerSettings()
}
.frame(minHeight: 800)
.injectFixtureEnvironmentObjects()
2021-11-04 22:01:27 +00:00
}
2021-09-25 08:18:22 +00:00
}