Нет AVPlayer Делегат?Как отследить, когда песня закончилась?Цель C разработка iPhone - PullRequest
83 голосов
/ 27 июля 2011

Я оглянулся, но не могу найти протокол делегата для AVPlayer class.Что дает?

Я использую его подкласс AVQueuePlayer для воспроизведения массива AVPlayerItems, каждый из которых загружается с URL-адреса.Можно ли как-нибудь вызвать метод, когда песня заканчивает играть?Особенно в конце очереди?

И если это невозможно, могу ли я как-нибудь вызвать метод, когда песня НАЧИНАЕТСЯ играть после буферизации?Я пытаюсь получить значок загрузки там, но он выключает значок до того, как на самом деле начинается музыка, даже если после действия [audioPlayer play].

Ответы [ 5 ]

149 голосов
/ 26 сентября 2012

Да, класс AVPlayer не имеет протокола делегирования, как AVAudioPlayer. Вам необходимо подписаться на уведомления на AVPlayerItem. Вы можете создать AVPlayerItem, используя тот же URL, который вы в противном случае передали бы -initWithURL: в AVPlayer.

-(void)startPlaybackForItemWithURL:(NSURL*)url {

    // First create an AVPlayerItem
    AVPlayerItem* playerItem = [AVPlayerItem playerItemWithURL:url];

    // Subscribe to the AVPlayerItem's DidPlayToEndTime notification.
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(itemDidFinishPlaying:) name:AVPlayerItemDidPlayToEndTimeNotification object:playerItem];

    // Pass the AVPlayerItem to a new player
    AVPlayer* player = [[[AVPlayer alloc] initWithPlayerItem:playerItem] autorelease];

    // Begin playback
    [player play]

} 

-(void)itemDidFinishPlaying:(NSNotification *) notification {
    // Will be called when AVPlayer finishes playing playerItem
}
8 голосов
/ 16 сентября 2012

Да.Добавьте наблюдателя KVO к статусу или рейтингу игрока:

- (IBAction)go {
   self.player = .....
   self.player.actionAtItemEnd = AVPlayerActionStop;
   [self.player addObserver:self forKeyPath:@"rate" options:0 context:0]; 
}

- (void)stopped {
    ...
    [self.player removeObserver:self]; //assumes we are the only observer
}

- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
    if (context == 0) {
        if(player.rate==0.0) //stopped
            [self stopped];
    }
    else
        [super observeVal...];
}

Итак, вот и все.

Отказ от ответственности: я написал здесь, чтобы не проверять, хороший ли код,ТАКЖЕ я никогда раньше не использовал AVPlayer, но это должно быть правильно.

5 голосов
/ 28 апреля 2017

Swift 3 - я добавляю наблюдателя в AVPlayerItem каждый раз, когда добавляю видео в плеер:

func playVideo(url: URL) {
  let playerItem = AVPlayerItem(asset: AVURLAsset(url: someVideoUrl))
  NotificationCenter.default.addObserver(self, selector: #selector(playerItemDidPlayToEndTime), name: NSNotification.Name.AVPlayerItemDidPlayToEndTime, object: playerItem)
  self.player.replaceCurrentItem(with: playerItem)
  self.player.play()
}

func playerItemDidPlayToEndTime() {
  // load next video or something
}
1 голос
/ 31 октября 2013

Я использую этот, и он работает:

_player = [[AVPlayer alloc]initWithURL:[NSURL URLWithString:_playingAudio.url]];
CMTime endTime = CMTimeMakeWithSeconds(_playingAudio.duration, 1);
timeObserver = [_player addBoundaryTimeObserverForTimes:[NSArray arrayWithObject:[NSValue valueWithCMTime:endTime]] queue:NULL usingBlock:^(void) {
    [_player removeTimeObserver:timeObserver];
    timeObserver = nil;
    //TODO play next sound
}];
[self play];

, где _playingAudio - мой пользовательский класс с некоторыми свойствами, а timeObserver - id ivar.

1 голос
/ 27 июля 2011

В документации Apple содержится много информации Руководство по программированию AVFoundation (см. Раздел «Воспроизведение мониторинга»).Похоже, что это в основном через KVO, так что вы можете освежить в памяти это, если вы не слишком знакомы (для этого есть руководство к Key Value Observing ProgrammingGuide .

...