Initial commit: SimVision tvOS streaming app

Features:
- VOD library with movie grouping and version detection
- TV show library with season/episode organization
- TMDB integration for trending shows and recently aired episodes
- Recent releases section with TMDB release date sorting
- Watch history tracking with continue watching
- Playlist caching (12-hour TTL) for offline support
- M3U playlist parsing with XStream API support
- Authentication with credential storage

Technical:
- SwiftUI for tvOS
- Actor-based services for thread safety
- Persistent caching for playlists, TMDB data, and watch history
- KSPlayer integration for video playback

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
2026-01-21 22:12:08 -06:00
commit 872354b834
283 changed files with 338296 additions and 0 deletions

View File

@@ -0,0 +1,98 @@
# AVPlayer踩坑记
AVPlayer是苹果平台上常用的视频播放器组件。使用简单性能好。
## 一、缓冲状态判断
有以下三个属性判断视频能否进行播放
```swift
isPlaybackBufferEmpty //
isPlaybackLikelyToKeepUp //
isPlaybackBufferFull //
```
一开始的写法是:
```swift
if playerItem.isPlaybackLikelyToKeepUp {
self.loadState = .playable
}
if playerItem.isPlaybackBufferEmpty {
self.loadState = .loading
}
```
后来发现有些高码率的视频,会出现一直加载,无法播放的情况。后来发现是因为缓存满了`isPlaybackBufferFull`的值为true, 但是`isPlaybackLikelyToKeepUp`的值还是为false。
所以改成是下面的写法:
```swift
if playerItem.isPlaybackLikelyToKeepUp || playerItem.isPlaybackBufferFull {
self.loadState = .playable
}
if playerItem.isPlaybackBufferEmpty {
self.loadState = .loading
}
```
最后发现有些视频会频繁的切换状态,分析了之后。发现 `isPlaybackBufferFull` `isPlaybackBufferEmpty`会同时为true。从字面上理解这两个值应该是要互斥的但是结果是相反。
所以最后正确的写法应该是:
```swift
if playerItem.isPlaybackBufferEmpty {
self.loadState = .loading
} else if playerItem.isPlaybackLikelyToKeepUp || playerItem.isPlaybackBufferFull {
self.loadState = .playable
}
```
## 二、视频播放黑屏
有些mp4视频在AVPlayer播放有声音但是画面黑屏。但是放在自研播放器`MEPlayer`就可以有声音又有画面了。所以解决方案就是能判断异常视频,并自动切换到`MEPlayer`
通过分析视频流发现帧率是25视频帧的格式是yuv444p。猜测应该是无法硬解yuv444p。所以导致视频帧无法显示后来通过增加AVPlayerItemOutput发现确实没有输出视频帧。
通过对每个属性进行尝试发现可以通过assetTrack的isPlayable属性来判断是否能输出视频帧
代码如下:
```swift
let videoTrack = item.tracks.first { $0.assetTrack?.mediaType.rawValue == AVMediaType.video.rawValue }
if let videoTrack = videoTrack, videoTrack.assetTrack?.isPlayable == false {
error = NSError(domain: AVFoundationErrorDomain, code: -1, userInfo: [NSLocalizedDescriptionKey: "can't player"])
return
}
```
## 三、多音轨
AVPlayer播放多音轨的时候。默认所有的音轨都会打开的。所以要改成只打开一个音轨
代码如下:
```swift
//
item.tracks.filter { $0.assetTrack?.mediaType.rawValue == AVMediaType.audio.rawValue }.dropFirst().forEach { $0.isEnabled = false }
```
## 四、AVQueuePlayer
AVQueuePlayer是AVPlayer的子类主要是用来实现视频顺序播放。也可以搭配AVPlayerLooper实现视频循环播放。
AVQueuePlayer默认是播放完一个视频就会自动切换到下个视频了。
这样就会导致一个问题,如果只是播放单个资源的话。那视频播放播放完之后,你就无法从头开始播放了。因为`currentItem`这个属性被重置为nil了。
这时你可以通过修改属性`actionAtItemEnd`的值。来改变视频播放完之后的行为。默认是`.advance`。直接改成是.`pause`或是`.none`就可以了
## 五、广告时间
以上所说的坑都已经在 [KSPlayer](https://git.code.oa.com/kintanwang/KSPlayer) 填了。KSPlayer是一款基于 AVPlayer, FFmpeg 纯Swift的音视频播放器支持所有视频格式和全景视频支持苹果全平台。实现高可用高性能的音视频播放能力。它包含UI控件模块、字幕模块、播放器内核模块。这些模块都是解耦的可以通过pod按需接入。欢迎大家试用。
#