Skip to content

jackdao1992/HysteriaPlayer

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Hysteria Player

HysteriaPlayer provides useful basic player functionalities.

It provides:

  • PlayerItem cache management.
  • Pre-buffer next PlayerItem.

Features:

  • You don't need to write KVO again, setting up few blocks then you can handle player status.
  • Ability to play previous PlayerItem.
  • If player suspended bacause of high network latency in bad network, auto-resume the playback of your PlayerItem when buffered ready.
  • Background playable enabled. (need to register your App supports background modes as "App plays audio")
  • Using getHysteriaOrder: to get the index of your PlayerItems.
  • Extends long time buffering in background.
  • Returns playing item's current and duration timescale.
  • PlayModes: Repeat, RepeatOne, Shuffle.

Installation

CocoaPods

If you using CocoaPods, it's easy to install HysteriaPlayer.

Podfile:

platform :ios, 'x.0'

pod 'HysteriaPlayer',			        '~> x.x.x'
    
end

Manually import library to your project

Add Frameworks

Add CoreMedia.framework, AudioToolbox.framework and AVFoundation.framework to your Link Binary With Libraries.

Copy provided point1sec.mp3 file to your Supporting Files

Ability to play the first PlayerItem when your application is resigned active but first PlayerItem is still buffering.

Register your app's background modes

Click your project and select your target app, going to the info tab find Required background modes , if not exist create new one. In Required background modes's item 0 copy this string App plays audio into it.

How to use - Setup

Register Handlers of HysteriaPlayer, all Handlers are optional.

#import "HysteriaPlayer.h"

...

- (void)initPlayer
{
    HysteriaPlayer *hysteriaPlayer = [HysteriaPlayer sharedInstance];
    
    [hysteriaPlayer registerHandlerPlayerRateChanged:^{
        // It will be called when player's rate changed, probely 1.0 to 0.0 or 0.0 to 1.0.
        // Anyways you should update your interface to notice the user what's happening. HysteriaPlayer have HysteriaPlayerStatus state helping you find out the informations.
        
        [self syncPlayPauseButtons];
    } CurrentItemChanged:^(AVPlayerItem *item) {
        // It will be called when player's currentItem changed.
        // If you have UI elements related to Playing item, should update them when called.(i.e. title, artist, artwork ..)
        
        [self syncPlayPauseButtons];
    } PlayerDidReachEnd:^{
        // It will be called when player stops, reaching the end of playing queue and repeat is disabled.
        
        UIAlertView *alert = [[UIAlertView alloc]initWithTitle:@"Player did reach end."
                                                       message:nil
                                                      delegate:self
                                             cancelButtonTitle:@"OK"
                                             otherButtonTitles:nil, nil];
        [alert show];
    }];
    
    [hysteriaPlayer registerHandlerCurrentItemPreLoaded:^(CMTime time) {
        // It will be called when current item receive new buffered data.
        
        NSLog(@"item buffered time: %f",CMTimeGetSeconds(time));
    }];
    
    [hysteriaPlayer registerHandlerReadyToPlay:^(HysteriaPlayerReadyToPlay identifier) {
        switch (identifier) {
            case HysteriaPlayerReadyToPlayPlayer:
                // It will be called when Player is ready to play at the first time.
                
                // If you have any UI changes related to Player, should update here.
                break;
            
            case HysteriaPlayerReadyToPlayCurrentItem:
                // It will be called when current PlayerItem is ready to play.
                
                // HysteriaPlayer will automatic play it, if you don't like this behavior,
                // You can pausePlayerForcibly:YES to stop it.
                break;
            default:
                break;
        }
    }];
    
    [hysteriaPlayer registerHandlerFailed:^(HysteriaPlayerFailed identifier, NSError *error) {
        switch (identifier) {
            case HysteriaPlayerFailedPlayer:
                break;
                
            case HysteriaPlayerFailedCurrentItem:
                // Current Item failed, advanced to next.
                [hysteriaPlayer playNext];
                break;
            default:
                break;
        }
        NSLog(@"%@", [error localizedDescription]);
    }];
}

Using initWithHandlers:

With callback blocks, handling the Player when status changed. All blocks are optional, set nil if you won't do anything on that callback block.

  • PlayerReadyToPlay : It will be called when Player is ready to play the PlayerItem, so play it. If you have play/pause buttons, should update their status after you starting play.

  • PlayerRateChanged : It will be called when player's rate changed, probely 1.0 to 0.0 or 0.0 to 1.0. Anyways you should update your interface to notice the user what's happening. HysteriaPlayer have HysteriaPlayerStatus state helping you find out the informations.

    • HysteriaPlayerStatusPlaying : Player is playing
    • HysteriaPlayerStatusForcePause : Player paused when Player's property PAUSE_REASON_ForcePause = YES.
    • HysteriaPlayerStatusBuffering : Player suspended because of no buffered.
    • HysteriaPlayerStatusUnknown : Player status unknown.
  • CurrentItem Changed : It will be called when player's currentItem changed. If you have UI elements related to Playing item, should update them when called.(i.e. title, artist, artwork ..)

  • ItemReadyToPlay : It will be called when current PlayerItem is ready to play.

  • PlayerPreLoaded : It will be called when receive new buffer data.

  • PlayerFailed : It will be called when player just failed.

  • PlayerDidReachEnd : It will be called when player stops, reaching the end of playing queue and repeat is disabled.

Setting up Source Getter

Before you starting play anything, you have to set up your data source for HysteriaPlayer. When Player gonna use (instantly play or pre-buffer) items, the source getter block will telling which index of your playing list is needed.

There are two methods to set up Source Getter.

  1. setupSourceGetter:ItemsCount:
  2. asyncSetupSourceGetter:ItemsCount:

ItemsCount tells HysteriaPlayer the counts of your data source, you have to update it using setItemsCount:(NSUInteger)count if your datasource's count is changed.

setupSourceGetter:ItemsCount:

The simplest way.
When player ask for an index that it would liked to use, return your source link as NSString value inside the index given block.

example:

    [hysteriaPlayer setupSourceGetter:^NSString *(NSUInteger index) {
        return [mp3Array objectAtIndex:index];
    } ItemsCount:[mp3Array count]];

asyncSetupSourceGetter:ItemsCount:

For advanced usage, if you could use setupSourceGetter:ItemsCount: as well then no needs to use this method to setting up.

If you have to access your media link when player actually gonna play that item. You probability take that media link by an asynchronous connection and HysteriaPlayer also needs an asynchronous block to transform the media link your provided to AVPlayerItem. There are no ways you can return values from an async block to another.

So, you have to call setupPlayerItem:Order: by yourself when your async connection that getting media link is completion. And the Order parameter is what player asked for.

example:

NSUInteger count = [listItems count];
[hysteriaPlayer asyncSetupSourceGetter:^(NSUInteger index) {
    asyncOperation^{
        ..
        operation
        ..
        NSString *mediaLink = source;
        
        [hysteriaPlayer setupPlayerItem:mediaLink Order:index];
    }
} ItemsCount:count];

Before you start

[HysteriaPlayer sharedInstance] will take over the audio focus, because it really init a Player for you.

The player object is singleton, you don't have to store it to your local instance, use [HysteriaPlayer sharedInstance] until you really need a Player to play anything.

It will be tiresome that your app take over the system audio focus when it just launched.

Snippets

Get item's index of my working items:

HysteriaPlayer *hysteriaPlayer = [HysteriaPlayer sharedInstance];
NSNumber *order = [hysteriaPlayer getHysteriaOrder:[hysteriaPlayer getCurrentItem]];

Get playing item's timescale

HysteriaPlayer *hysteriaPlayer = [HysteriaPlayer sharedInstance];
NSDictionary *dict = [hysteriaPlayer getPlayerTime];
double durationTime = [[dict objectForKey:@"DurationTime"] doubleValue];
double currentTime = [[dict objectForKey:@"CurrentTime"] doubleValue];

How to pause my playback forcibly?

pausePlayerForcibly:(BOOL) method telling HysteriaPlayer to/not to force pause the playback(mostly when user tapped play/pause button)

- (IBAction)play_pauseButton:(id)sender
{
    HysteriaPlayer *hysteriaPlayer = [HysteriaPlayer sharedInstance];
    
    if ([hysteriaPlayer isPlaying])
    {
        [hysteriaPlayer pausePlayerForcibly:YES];
        [hysteriaPlayer pause];
    }else{
        [hysteriaPlayer pausePlayerForcibly:NO];
        [hysteriaPlayer play];
    }
}

Get Player status

switch ([hysteriaPlayer getHysteriaPlayerStatus]) {
    case HysteriaPlayerStatusUnknown:
        
        break;
    case HysteriaPlayerStatusForcePause:
        
        break;
    case HysteriaPlayerStatusBuffering:
        
        break;
    case HysteriaPlayerStatusPlaying:
        
    default:
        break;
}

Disable played item caching

Default is cache enabled

HysteriaPlayer *hysteriaPlayer = [HysteriaPlayer sharedInstance];
[hysteriaPlayer enableMemoryCached:NO];

What if I don't need player instance anymore?

HysteriaPlayer *hysteriaPlayer = [HysteriaPlayer sharedInstance];
[hysteriaPlayer deprecatePlayer];
hysteriaPlayer = nil;

Licenses

All source code is licensed under the MIT License.

Author

Created by Saiday

Analytics Bitdeli Badge

About

Objective-C audio player, sitting on top of AVPlayer

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • Objective-C 97.5%
  • Swift 2.2%
  • Ruby 0.3%