Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

iOS Audio Streaming only works for **SOME** bluetooth devices?

I am working on iOS app which will be compatible with iOS 6/7 and stream audio .mp3 files from a website.

I have already gotten this to work using the following code:

-(NSString*)documentsFolder
{
    NSString* dataPath =  [NSHomeDirectory() stringByAppendingPathComponent:@"Documents"];
    if (![[NSFileManager defaultManager] fileExistsAtPath:dataPath])
        [[NSFileManager defaultManager] createDirectoryAtPath:dataPath withIntermediateDirectories:NO attributes:nil error:NULL];
    return dataPath;
}

-(NSString*)createURLFile:(NSString*)songURL
{
   NSString* M3U_FILE = @"song.m3u";
   NSString* path = [NSString stringWithFormat:@"%@",[[self documentsFolder] stringByAppendingPathComponent:M3U_FILE]];
    if([[NSFileManager defaultManager] createFileAtPath:path contents:nil attributes:nil])
    {
      NSFileHandle* outFile = [NSFileHandle fileHandleForWritingAtPath:path];
        if(outFile != nil)
      {
         NSData* buffer = [songURL dataUsingEncoding:NSUTF8StringEncoding];
         [outFile writeData:buffer];
         return path;
      }
   }
    return nil;
}


- (void)createStreamer
{
    // Remove any previous references.
    [[NSNotificationCenter defaultCenter] removeObserver:self];
    // Create a new player.
    NSString* fileURL = [self createURLFile:self.aSong.songpath];
    self.songPlayer = [[AVPlayer alloc]initWithURL:[NSURL fileURLWithPath:fileURL]];
    NSAssert(self.songPlayer != nil, @"NIL AVPlayer Created!!!");
    // Observer for when the song ends...
    [[NSNotificationCenter defaultCenter] addObserver:self
                                             selector:@selector(playerItemDidReachEnd:)
                                                 name:AVPlayerItemDidPlayToEndTimeNotification
                                               object:[self.songPlayer currentItem]];
    [[UIApplication sharedApplication] setIdleTimerDisabled: YES];
}

I store the url for the .mp3 file in a local .m3u file and use that to load up the AVPlayer. In earlier versions of iOS, I was told that the AVPlayer would load the song first and then play it, not stream it immediately. While this does not appear to be true in iOS 6/7 (the song starts streaming almost immediately), the .m3u file was being created in case there were any problems created by not having it done this way.

With this, a loop is monitoring the status of the AVPlayer and after a few seconds, the audio starts to play out the phone without a problem.

For testing purposes, I set up an MPVolumeView on the page which plays songs:

MPVolumeView *volumeView = [[[MPVolumeView alloc] initWithFrame:CGRectMake(0, 0, 310, 20)] autorelease];
volumeView.center = CGPointMake(160,62);
[volumeView sizeToFit];
[self.view addSubview:volumeView];

The reason for this is that the volume slider will also show an indicator if the bluetooth is connected as an audio output source and allow me to change the audio route between the phone and the bluetooth device. So far, so good.

I connected my phone to my Jawbox Jambone via bluetooth, start the AVPlayer on a song, and the song comes out of the Jawbox as expected. The volume control has the small "rectangle with arrow" indicating that I can switch the audio output and indeed, while the song is playing, I can switch between the phone and the Jawbox. Happiness.

The problem occurs when I try to connect it to a car. I have two experiences with this:

  1. The car is already paired with the phone for making/receiving calls. The phone even indicates it is paired when I get into the car. But when I use the same code to play the same audio files, they only come out of the phone. The volume slider does not show the "bluetooth route" indicator at all (like it does not recognize the car as a audio output route).
  2. In another car, the audio was streaming from another app (some radio streaming app). The other app was stopped and this one started. The audio started playing for the same song tested above, but stopped after a second or two. Again, there was no indicator on the volume slider that the bluetooth was connected at this point.

Can somebody explain to me why the audio could stream fine out to one bluetooth device but not to another?

Have I missed something (an entitlement?) in the profile for my app that will allow it to stream audio via bluetooth to a car?

like image 365
FuzzyBunnySlippers Avatar asked Jan 18 '14 12:01

FuzzyBunnySlippers


People also ask

Where is advanced Bluetooth settings on iPhone?

How to access Bluetooth controls from Control Center in iOS 13. Swipe up from the bottom of your iPhone screen, or down from the upper-right corner of an iPhone X or later. Long press on the section of Control Center with wireless controls like Airplane Mode, Bluetooth, and Wi-Fi to open the broader wireless controls.

Is streaming audio the same as Bluetooth?

Both Bluetooth and Wi-Fi streaming use license-free frequencies to transmit audio. However, the setup and the connection of the data streams have completely different ways of working: With Bluetooth loudspeakers, data is always exchanged directly between the transmitter and receiver.

Why is my iPhone not picking up Bluetooth devices?

Make sure that your Bluetooth accessory and iOS or iPadOS device are close to each other. Turn your Bluetooth accessory off and back on again. Make sure that your Bluetooth accessory is on and fully charged or connected to power. If your accessory uses batteries, see if they need to be replaced.


2 Answers

There is this project at GIT. Play iOS project is a streaming client for Play that runs on your iPhone/iPad. It supports background audio as well as the media keys when backgrounded. It supports:

  • Streams shoutcast stream
  • Displays currently playing track
  • Background audio
  • Lock screen album art & play controls
  • AirPlay (along with Bluetooth) streaming. Supports sending metadata and album art

You can download the project here. I have not tested this on CAR bluetooth audio player though. Hope it may be of any help to you.

like image 81
Aditya Avatar answered Sep 30 '22 14:09

Aditya


In the first example, your car may simply be a remote player. You would need to register for remote events like this (consider using an AVAudioPlayer instead of an AVPlayer also)

Setup the AudioSession to recognize a bluetooth audio route:

- (BOOL)prepareAudioSession {

    // deactivate existing session
NSError *setCategoryError = nil;
NSError *activationError = nil;

BOOL success = [[AVAudioSession sharedInstance] setActive:NO error: nil];
if (!success) {
    NSLog(@"deactivationError");
}

    // set audio session category AVAudioSessionCategoryPlayAndRecord options AVAudioSessionCategoryOptionAllowBluetooth
success = [[AVAudioSession sharedInstance] setCategory:AVAudioSessionCategoryPlayback withOptions:AVAudioSessionCategoryOptionAllowBluetooth error:&setCategoryError];
if (!success)
{
    NSLog(@"setCategoryError %@",setCategoryError);
}

    // activate audio session
success = [[AVAudioSession sharedInstance] setActive:YES error: &activationError];
if (!success) {
    NSLog(@"activationError");
}

return success;

}

You can check the routes:

 AVAudioSessionRouteDescription *mAVASRD = audioSession.currentRoute;
NSLog(@"the array is %@",mAVASRD.outputs);

for (int ctr = 0; ctr < [mAVASRD.outputs count]; ctr++)
{
    AVAudioSessionPortDescription *myPortDescription = [mAVASRD.outputs objectAtIndex:ctr];
    NSLog(@"the type is %@",myPortDescription.portType);
    NSLog(@"the name is %@",myPortDescription.portName);
    NSLog(@"the UID is %@",myPortDescription.UID);
    NSLog(@"the data sources are %@",myPortDescription.dataSources);
}

Then initialize your AVAudioPlayer and turn on RemoteControlEvents (you can use the console in your car to send play/pause/etc)

[[UIApplication sharedApplication] beginReceivingRemoteControlEvents];

then implement something like the delegate method for AVAudioPlayer in this stack overflow question to capture the received events and react accordingly in your code:

AVAudioPlayer on Lock Screen

In the scenario 2, when you moved one app to the background (the radio streaming app) and started your app, the likely culprit for the issue is the same cause - your app has to recognize the bluetooth route for audio.

By the way for phone calls and Siri, the iOS uses a different Bluetooth channel that the default for remote control (which is the one I am describing for your car).

When you setup this route and remote control events, you also get a bonus byproduct - your app will be controllable from the lock screen. Check out this technical note from Apple to configure your app to play in the background as well if that is also something you need to do when the screen locks: Technical QA document QA1668

Finally, for added integration via your bluetooth route, look at MPNowPlayingInfoCenter - put the title artist artwork and other good stuff on the lock screen and on most bluetooth screens in the car that are displaying that information.

like image 33
thebdog Avatar answered Sep 30 '22 15:09

thebdog