我正在播放一个电视节目,该节目已经使用AVQueuePlayer切换到我项目的不同章节。我还想提供跳过上一章/下一章的可能性,或者在AVQueuePlayer播放时动态选择不同的章节。
跳过下一个项目对于AVQueuePlayer提供的advanceToNextItem
没有问题,但是从队列中跳回或播放特定项目没有任何相似之处。
所以我不太确定这里最好的方法是什么:
replaceCurrentItemWithPlayerItem:
调用actionAtItemEnd
来播放nextItem并使用'replaceCurrentItemWithPlayerItem'让用户选择某个章节要么
附加信息:我按照它们应该出现在NSArray中的顺序存储不同视频的路径用户应该通过按下代表章节的按钮跳转到某些章节。按钮有标签,也是阵列中相应视频的索引。
希望我能说清楚吗?有这种情况经验的人吗?如果有人知道在哪里买一个提供功能的好的IOS VideoPlayerFramework,我也很感激这个链接。
如果您希望您的程序可以播放上一项并从您的播放项目(NSArray)播放所选项目,则可以执行此操作。
- (void)playAtIndex:(NSInteger)index
{
[player removeAllItems];
for (int i = index; i <playerItems.count; i ++) {
AVPlayerItem* obj = [playerItems objectAtIndex:i];
if ([player canInsertItem:obj afterItem:nil]) {
[obj seekToTime:kCMTimeZero];
[player insertItem:obj afterItem:nil];
}
}
}
编辑:playerItems是存储AVPlayerItems的NSMutableArray(NSArray)。
第一个答案从AVQueuePlayer中删除所有项目,并从作为索引arg传递的迭代开始重新填充队列。这将启动具有前一项的新填充队列(假设您传递了正确的索引)以及从该点开始的现有playerItems数组中的其余项目,但是它不允许多次反转,例如,你正在第10轨,并希望返回并重播赛道9,然后重播赛道5,上面你无法完成。但在这里你可以......
-(IBAction) prevSongTapped: (id) sender
{
if (globalSongCount>1){ //keep running tally of items already played
[self resetAVQueue]; //removes all items from AVQueuePlayer
for (int i = 1; i < globalSongCount-1; i++){
[audioQueuePlayer advanceToNextItem];
}
globalSongCount--;
[audioQueuePlayer play];
}
}
以下代码允许您跳转到您的任何项目。没有球员前进。干净利落。 playerItemList是带有AVPlayerItem对象的NSArray。
- (void)playAtIndex:(NSInteger)index
{
[audioPlayer removeAllItems];
AVPlayerItem* obj = [playerItemList objectAtIndex:index];
[obj seekToTime:kCMTimeZero];
[audioPlayer insertItem:obj afterItem:nil];
[audioPlayer play];
}
@ saiday的回答对我有用,这里是他答案的快捷版本
func play(at index: Int) {
queue.removeAllItems()
for i in index..<items.count {
let obj: AVPlayerItem? = items[i]
if queue.canInsert(obj!, after: nil) {
obj?.seek(to: kCMTimeZero, completionHandler: nil)
queue.insert(obj!, after: nil)
}
}
}
这应该是AVQueuePlayer对象的责任,而不是视图控制器本身,因此您应该使其可重用并通过扩展公开其他答案实现,并以与advanceToNextItem()类似的方式使用它:
extension AVQueuePlayer {
func advanceToPreviousItem(for currentItem: Int, with initialItems: [AVPlayerItem]) {
self.removeAllItems()
for i in currentItem..<initialItems.count {
let obj: AVPlayerItem? = initialItems[i]
if self.canInsert(obj!, after: nil) {
obj?.seek(to: kCMTimeZero, completionHandler: nil)
self.insert(obj!, after: nil)
}
}
}
}
用法(您只需存储索引和对初始队列播放器项的引用):
self.queuePlayer.advanceToPreviousItem(for: self.currentIndex, with: self.playerItems)
维护索引的一种方法是观察每个视频项的AVPlayerItemDidPlayToEndTime通知:
func addDidFinishObserver() {
queuePlayer.items().forEach { item in
NotificationCenter.default.addObserver(self, selector: #selector(playerDidFinishPlaying), name: Notification.Name.AVPlayerItemDidPlayToEndTime, object: item)
}
}
func removeDidFinishObserver() {
queuePlayer.items().forEach { item in
NotificationCenter.default.removeObserver(self, name: NSNotification.Name.AVPlayerItemDidPlayToEndTime, object: item)
}
}
@objc func playerDidFinishPlaying(note: NSNotification) {
if queuePlayer.currentItem == queuePlayer.items().last {
print("last item finished")
} else {
print("item \(currentIndex) finished")
currentIndex += 1
}
}
这种观察对于其他用例(进度条,当前视频计时器重置......)也非常有用。
如果你想使用AVQueuePlayer从任何索引播放歌曲。那么下面的代码可以帮助你。
NSMutableArray *musicListarray (add song that you want to play in queue);
AVQueuePlayer *audioPlayer;
AVPlayerItem *item;
-(void) nextTapped
{
nowPlayingIndex = nowPlayingIndex + 1;
if (nowPlayingIndex > musicListarray.count)
{
}
else
{
[self playTrack];
}
}
-(void) playback
{
if (nowPlayingIndex < 0)
{
}
else
{
nowPlayingIndex = nowPlayingIndex - 1;
[self playTrack];
}
}
-(void) playTrack
{
@try
{
if (musicArray.count > 0)
{
item =[[AVPlayerItem alloc] initWithURL: [NSURL URLWithString:musicListarray
[nowPlayingIndex]]];
[audioPlayer replaceCurrentItemWithPlayerItem:item];
[audioPlayer play];
}
}
@catch (NSException *exception)
{
}
}
-(void) PlaySongAtIndex
{
//yore code...
nowPlayingIndex = i (stating index from queue)[audioPlayer play];
[self playTrack];
}
当你想播放一首歌时,在这里播放PlaySongAtIndex。