Android can't record video with Front Facing Camera, MediaRecorder start failed: -19
I wrestled with this problem a bit today, too.
First, make sure that your permissions are set up correctly. Specifically, to record video, you'll want:
<uses-feature android:name="android.hardware.camera.front" />
<uses-feature android:name="android.hardware.microphone"/>
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.RECORD_AUDIO" />
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
Second, and this is the tricky part, this line from the tutorial does not work with the front-facing camera!
mMediaRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH));
That signature for CamcorderProfile.get() defaults to a profile for the back-facing camera:
Returns the camcorder profile for the first back-facing camera on the device at the given quality level. If the device has no back-facing camera, this returns null.
Instead, use http://developer.android.com/reference/android/media/CamcorderProfile.html#get(int,%20int). The first parameter is the id of the camera that you opened, specifically, the front-facing camera.
Okay so I finally have it working sort of.
The issue seems to definitely relate to Profile Settings
and in particular Frame Rate
.
On the Nexus S, my primary test device, if I probe the Camera I receive following Parameters:
For the Rear Camera:
15 FPS to 30 FPS, fair enough.
For the Front Facing Camera:
7.5 FPS to 30 FPS, okay.
Then I check the Profiles I am trying to use:
CamcorderProfile.QUALITY_HIGH
CamcorderProfile.QUALITY_LOW
QUALITY_LOW:
audioBitRate: 12200
audioChannels: 1
audioCodec: AMR_NB audioSampleRate: 8000
duration: 30
fileFormat: THREE_GPP
quality: 0
videoBitRate: 256000
videoCodec: H264
videoFrameRate: 30
videoFrameWidth: 176
videoFrameHeight: 144
QUALITY_HIGH:
audioBitRate: 24000
audioChannels: 1
audioCodec: AAC
audioSampleRate: 16000
duration: 60
fileFormat: MPEG_4
quality: 1
videoBitRate: 3000000
videoCodec: H264
videoFrameRate: 30
videoFrameWidth: 720
videoFrameHeight: 480
Clearly, the High Quality Profile is meant for the Rear Camera, seeing as the front facing is only 640x480. But they both state 30 FPS.
Now.... Here's the weirdness:
If I set ANY frame rate for the rear facing camera, no matter what profile, it crashes with the dreaded:
-19 error
mediaRecorder.setVideoFrameRate(fpsInt);
That's not a big deal coz I don't care about the rear camera but it is weird, considering the profiles are defaulting to 30 and the Params
say they accept 15-30. But no int value I've tried has worked. If I omit the setVideoFrameRate
it's fine.
Anyway, moving onto the Front Facing Camera.
So, if I use the QUALITY_LOW
profile AND set the frame-rate to 15 or lower, it magically works.
mediaRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_LOW));
mediaRecorder.setVideoFrameRate(15);
In fact any value, 1 - 15 works. Which seems weird.
So here's the conundrum, I can probably probe for resolution and select an appropriate res for most cameras, although I'm also fairly confident almost all front-facing cameras at minimum VGA 640x480.
But, what about the frame rate? In the case of the Nexus S, I don't see any way I could determine the value of 15 or lower without just guessing? Should I aim to always use the LOWEST frame-rate that is returned by the Camera?
I took a look at the Galaxy Nexus and it has 3 frame rate ranges, the first one is 15 - 15 and the second is 15 - 30. Its low quality profile is similar albeit higher resolution. If I use low profile on Galaxy Nexus it seems to work fine.
With the Nexus 7, I cant probe the CamcorderProfile's I keep getting null pointers, which is weird.
It says it supports 4 FPS - 60 FPS.
If I choose QUALITY_LOW
which you'd think it should work, it crashes, and I can't find a frame rate it will work with.
Although the error relates to setProfile
, so I think the issue is with the built in profile.
Surely the point of Android API is that it's consistent, this is a flag ship device and the FF camera is there for Video Conferencing isn't it?????
So, while I have it working on two of the devices using manual custom settings for each, I can't see a clear way of making it work across multiple devices through code.
It seems that the Nexus S does not behave the way it promises to with regards to setting the FPS as per its Camera.getParameters().getSupportedPreviewFpsRange()
I'm all happy for it to use Auto FPS settings but apparently it won't with the FF camera so what am I supposed to do? I have to explicitly set the FPS on the Nexus S and in this case to anything from 1 to 15 FPS, despite the Camera telling me it handles 7.5 - 30 FPS.
Seems like the promise of the setProfile
fixing all the issues in 2.x wasn't entirely true.
I can understand if your writing the Camera App for a particular ROM you just customize it to that particular hardware, which might explain why people seem to always have buggy camera apps on custom ROMs. BUT..... How do downloadable video recording apps work? Are they custom to each device?
Is this why there's no Facebook Poke and Twitter Vine on Android yet???? :P
Google, what is with your Camera API?
Does anyone know the "best practices" to determine resolution and frame-rate for all API 15+
compatible devices?
Is that even possible, or am I going to be writing custom code on each device I test and then just roll the dice on the rest?
Or is the Nexus S and the Nexus 7 just freak accidents?
try with QUALITY_LOW because QUALITY_HIGH is not supported in Front Camera.