mirror of
https://github.com/yattee/yattee.git
synced 2024-11-10 00:08:21 +00:00
322a550666
- iPad: rotate to device orientation on startup - fixed controls in portrait fullscreen - iOS: don’t call setNeedsDrawing multiple times - On iOS we call set needs drawing only once. - Added cooldown time to MPV.Client setNeedsDrawing to avoid multiple successive calls - make fullscreen animation smoother - dragGesture now calls toggleFullScreenAction - fix tvOS and macOS build Signed-off-by: Toni Förster <toni.foerster@gmail.com>
499 lines
18 KiB
Swift
499 lines
18 KiB
Swift
import Defaults
|
|
import SwiftUI
|
|
|
|
struct PlayerSettings: View {
|
|
@Default(.instances) private var instances
|
|
@Default(.playerInstanceID) private var playerInstanceID
|
|
|
|
@Default(.playerSidebar) private var playerSidebar
|
|
|
|
@Default(.showKeywords) private var showKeywords
|
|
@Default(.showComments) private var showComments
|
|
#if !os(tvOS)
|
|
@Default(.showScrollToTopInComments) private var showScrollToTopInComments
|
|
@Default(.collapsedLinesDescription) private var collapsedLinesDescription
|
|
@Default(.exitFullscreenOnEOF) private var exitFullscreenOnEOF
|
|
#endif
|
|
@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
|
|
@Default(.closePlayerOnOpeningPiP) private var closePlayerOnOpeningPiP
|
|
#if !os(macOS)
|
|
@Default(.pauseOnEnteringBackground) private var pauseOnEnteringBackground
|
|
@Default(.closePiPAndOpenPlayerOnEnteringForeground) private var closePiPAndOpenPlayerOnEnteringForeground
|
|
#endif
|
|
|
|
@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
|
|
@Default(.captionsFallbackLanguageCode) private var captionsFallbackLanguageCode
|
|
@Default(.captionsFontScaleSize) private var captionsFontScaleSize
|
|
@Default(.captionsFontColor) private var captionsFontColor
|
|
|
|
@ObservedObject private var accounts = AccountsModel.shared
|
|
|
|
#if os(iOS)
|
|
private var idiom: UIUserInterfaceIdiom {
|
|
UIDevice.current.userInterfaceIdiom
|
|
}
|
|
#endif
|
|
|
|
#if os(tvOS)
|
|
@State private var isShowingDefaultLanguagePicker = false
|
|
@State private var isShowingFallbackLanguagePicker = false
|
|
#endif
|
|
|
|
var body: some View {
|
|
Group {
|
|
#if os(macOS)
|
|
sections
|
|
|
|
Spacer()
|
|
#else
|
|
List {
|
|
sections
|
|
}
|
|
#endif
|
|
}
|
|
#if os(tvOS)
|
|
.frame(maxWidth: 1000)
|
|
#elseif os(iOS)
|
|
.listStyle(.insetGrouped)
|
|
#endif
|
|
.navigationTitle("Player")
|
|
}
|
|
|
|
private var sections: some View {
|
|
Group {
|
|
Section(header: SettingsHeader(text: "Playback".localized())) {
|
|
if !accounts.isEmpty {
|
|
sourcePicker
|
|
}
|
|
pauseOnHidingPlayerToggle
|
|
closeVideoOnEOFToggle
|
|
#if os(macOS)
|
|
exitFullscreenOnEOFToggle
|
|
#endif
|
|
#if !os(macOS)
|
|
pauseOnEnteringBackgroundToogle
|
|
#endif
|
|
}
|
|
|
|
#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())) {
|
|
#if os(tvOS)
|
|
Text("Size").font(.subheadline)
|
|
#endif
|
|
captionsFontScaleSizePicker
|
|
#if os(tvOS)
|
|
Text("Color").font(.subheadline)
|
|
#endif
|
|
captionsFontColorPicker
|
|
showCaptionsAutoShowToggle
|
|
|
|
#if !os(tvOS)
|
|
captionDefaultLanguagePicker
|
|
captionFallbackLanguagePicker
|
|
#else
|
|
Button(action: { isShowingDefaultLanguagePicker = true }) {
|
|
HStack {
|
|
Text("Default language")
|
|
Spacer()
|
|
Text("\(LanguageCodes(rawValue: captionsDefaultLanguageCode)!.description.capitalized) (\(captionsDefaultLanguageCode))").foregroundColor(.secondary)
|
|
}
|
|
}
|
|
.frame(maxWidth: .infinity).sheet(isPresented: $isShowingDefaultLanguagePicker) {
|
|
defaultLanguagePickerTVOS(
|
|
selectedLanguage: $captionsDefaultLanguageCode,
|
|
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
|
|
)
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#if !os(tvOS)
|
|
Section(header: SettingsHeader(text: "Chapters".localized())) {
|
|
showChaptersToggle
|
|
showThumbnailsToggle
|
|
showThumbnailsWhenDifferentToggle
|
|
expandChaptersToggle
|
|
}
|
|
#endif
|
|
|
|
let interface = Section(header: SettingsHeader(text: "Interface".localized())) {
|
|
#if os(iOS)
|
|
if idiom == .pad {
|
|
sidebarPicker
|
|
}
|
|
#endif
|
|
|
|
#if os(macOS)
|
|
sidebarPicker
|
|
#endif
|
|
|
|
if !accounts.isEmpty {
|
|
keywordsToggle
|
|
|
|
commentsToggle
|
|
#if !os(tvOS)
|
|
showScrollToTopInCommentsToggle
|
|
#endif
|
|
|
|
returnYouTubeDislikeToggle
|
|
}
|
|
}
|
|
|
|
#if os(tvOS)
|
|
if !accounts.isEmpty {
|
|
interface
|
|
}
|
|
#elseif os(macOS)
|
|
interface
|
|
#elseif os(iOS)
|
|
if idiom == .pad || !accounts.isEmpty {
|
|
interface
|
|
}
|
|
#endif
|
|
|
|
#if os(iOS)
|
|
Section(header: SettingsHeader(text: "Fullscreen".localized())) {
|
|
if Constants.isIPad {
|
|
enterFullscreenInLandscapeToggle
|
|
}
|
|
|
|
exitFullscreenOnEOFToggle
|
|
rotateToLandscapeOnEnterFullScreenPicker
|
|
}
|
|
#endif
|
|
|
|
Section(header: SettingsHeader(text: "Picture in Picture".localized())) {
|
|
closePiPOnNavigationToggle
|
|
closePiPOnOpeningPlayerToggle
|
|
closePlayerOnOpeningPiPToggle
|
|
#if !os(macOS)
|
|
closePiPAndOpenPlayerOnEnteringForegroundToggle
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
private var videoDetailsHeaderPadding: Double {
|
|
#if os(macOS)
|
|
5.0
|
|
#else
|
|
0.0
|
|
#endif
|
|
}
|
|
|
|
private var sourcePicker: some View {
|
|
Picker("Source", selection: $playerInstanceID) {
|
|
Text("Instance of current account").tag(String?.none)
|
|
|
|
ForEach(instances) { instance in
|
|
Text(instance.description).tag(Optional(instance.id))
|
|
}
|
|
}
|
|
.modifier(SettingsPickerModifier())
|
|
}
|
|
|
|
private var sidebarPicker: some View {
|
|
Picker("Sidebar", selection: $playerSidebar) {
|
|
#if os(macOS)
|
|
Text("Show sidebar").tag(PlayerSidebarSetting.always)
|
|
#endif
|
|
|
|
#if os(iOS)
|
|
Text("Show sidebar when space permits").tag(PlayerSidebarSetting.whenFits)
|
|
#endif
|
|
|
|
Text("Hide sidebar").tag(PlayerSidebarSetting.never)
|
|
}
|
|
.modifier(SettingsPickerModifier())
|
|
}
|
|
|
|
private var commentsToggle: some View {
|
|
Toggle("Show comments", isOn: $showComments)
|
|
}
|
|
|
|
#if !os(tvOS)
|
|
private var showScrollToTopInCommentsToggle: some View {
|
|
Toggle("Show scroll to top button in comments", isOn: $showScrollToTopInComments).disabled(!showComments)
|
|
}
|
|
#endif
|
|
|
|
private var keywordsToggle: some View {
|
|
Toggle("Show keywords", isOn: $showKeywords)
|
|
}
|
|
|
|
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")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
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)
|
|
}
|
|
|
|
#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)
|
|
}
|
|
|
|
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
|
|
|
|
private var showCaptionsAutoShowToggle: some View {
|
|
Toggle("Always show captions", isOn: $captionsAutoShow)
|
|
}
|
|
|
|
private var captionsFontScaleSizePicker: some View {
|
|
Picker("Size", selection: $captionsFontScaleSize) {
|
|
Text("Small").tag(String("0.725"))
|
|
Text("Medium").tag(String("1.0"))
|
|
Text("Large").tag(String("1.5"))
|
|
}
|
|
.onChange(of: captionsFontScaleSize) { _ in
|
|
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)
|
|
}
|
|
#if os(macOS)
|
|
.labelsHidden()
|
|
#endif
|
|
}
|
|
|
|
#if !os(tvOS)
|
|
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
|
|
}
|
|
|
|
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
|
|
}
|
|
#else
|
|
struct defaultLanguagePickerTVOS: 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 Default Language")
|
|
}
|
|
}
|
|
}
|
|
|
|
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")
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#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
|
|
}
|
|
|
|
struct PlayerSettings_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
VStack(alignment: .leading) {
|
|
PlayerSettings()
|
|
}
|
|
.frame(minHeight: 800)
|
|
.injectFixtureEnvironmentObjects()
|
|
}
|
|
}
|