chapter
One of Camera development series – display Camera real-time picture
Camera Development series ii – Camera preview data callback
Camera development series 3 – Camera data hardcoded to H264
Camera development series iv – encapsulate encoded audio and video into mp4 container using MediaMuxer
Camera Development series 5 – Use MediaExtractor to make a simple player
Camera Development Series 6 – Video push streaming using mina framework
Recently, I have been doing android Camera functions. I have to say that there are many pits here. There are not too many places to pay attention to, which can be said to be from the entry to the ground series. Anyway, this thing almost killed me.
This series of articles uses the APIS under the Android.Hardware.Camera package, not Camer2. This is mainly for compatibility issues, but another important reason is that I haven’t seen the camera2 documentation yet and nothing is going to =_=.
This article is mainly divided into the following points:
- How to call the phone camera and get real-time data
- How to display the camera preview screen
- What to pay attention to when using the camera
Implement camera preview
Mainfest Permission statement
Before developing the device camera, you need to declare the following permissions in Mainfest:
<uses-permission android:name="android.permission.CAMERA" />
Copy the code
If your app needs to store photos or videos to the device, you must specify write permissions in the Manifest file:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
Copy the code
If recording is also required, add recording permission:
<uses-permission android:name="android.permission.RECORD_AUDIO" />
Copy the code
Of course, if you need to take photos to record the location, you also need to apply for the following permissions:
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
Copy the code
After you declare permissions, you can use the camera-related apis
Detection of camera hardware
Before you use a camera, what do you do first? Of course, it is necessary to check whether the device has the camera hardware first, not to mention your mobile phone, the application you make is not you alone use your heart do not have a 13 number? If there is camera hardware, then go further to access the camera. The following is a code example to check whether the camera hardware exists:
private boolean checkCameraHardware(Context context) { if(context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_CAMERA)){
return true;
} else {
return false; }}Copy the code
Android devices can have multiple camera hardware. Now, generally, mobile phones have two cameras in front and rear. The id of the rear camera is 0, and the ID of the front camera is 1. To access basic Camera functions, use the Camera’s open() method to get an instance of the Camera.
try {
Camera camera = Camera.open(mCamerId);
}catch (Exception e){
LogUtil.i("The camera is occupied.");
e.printStackTrace();
}
Copy the code
Remember to catch exceptions here, because it is possible that another application may crash the camera.
Gets & Sets camera parameters
Once you have successfully accessed the Camera device, you can use the Camera#getParameters() method to get Camera parameter information. You can view what parameter Settings are supported by the Camera. You can also setParameters for the camera using the Camera#setParameters method.
private void initCamera(int width,int height){
Camera.Parameters parameters = mCamera.getParameters();
parameters.setPreviewFormat(ImageFormat.NV21);
// Calculate the appropriate width and height algorithm according to the set width and height and the resolution supported by the phone
Camera.Size optionSize = CameraUtil
.getInstance(mCamera, mCamerId)
.getOptimalPreviewSize(width, height);
parameters.setPreviewSize(optionSize.width, optionSize.height);
// Set the photo size
parameters.setPictureSize(optionSize.width, optionSize.height);
// Set real-time focus. Some phones do not support crash
parameters.setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_PICTURE);
mCamera.setParameters(parameters);
// Enable preview
mCamera.startPreview();
}
Copy the code
Displays the camera preview screen
The camera is ready, but there is one last step. In order to take pictures or video effectively, we must have a preview of the camera on the screen. A camera preview class uses the SurfaceView control to display the preview data from the camera in real time, so that we can see each frame and capture images or videos.
mSurfaceView = findViewById(R.id.surfaceView);
mHolder = mSurfaceView.getHolder();
mHolder.addCallback(new SurfaceCallback());
Copy the code
What is SurfaceCallback? The SurfaceCallback inherits the SurfaceView.Callback interface class and needs to implement interface methods that listen for the creation of the SurfaceView control and the Callback of the destruction event associated with the camera preview display in the Callback method.
private class SurfaceCallback implements SurfaceHolder.Callback {
@Override
public void surfaceCreated(SurfaceHolder holder) {
try {
mCamera = Camera.open(mCamerId);
}catch (Exception e){
LogUtil.i("The camera is occupied."); e.printStackTrace(); }}@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
try {
initCamera(mSurfaceView.getWidth(),mSurfaceView.getHeight());
mCamera.setPreviewDisplay(holder);
} catch(Exception e) { e.printStackTrace(); }}@Override
public void surfaceDestroyed(SurfaceHolder holder) {}}Copy the code
The layout file is simple:
<?xml version="1.0" encoding="utf-8"? >
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<SurfaceView
android:id="@+id/surfaceView"
android:layout_width="match_parent"
android:layout_height="match_parent"
/>
</LinearLayout>
Copy the code
Now you can hit Run and you should see something like this:
Hey, what’s going on, how is the picture rotating, is there a problem with the code? Nonono, don’t panic, the default orientation of camera preview is landscape, so in this example the layout specifies the horizontal orientation and fixed the application to landscape. To make it easy to render a camera preview, you can specify the orientation of your Activity to landscape in the manifest configuration file.
<activity android:name=".MainActivity" android:screenOrientation="landscape">
Copy the code
You can also call this method before startPreview() is called using the following method:
/** * After you get the default camera rotation Angle, rotate it back to note that it is counterclockwise **@param activity
*/
public void setCameraDisplayOrientation(Activity activity) {
Camera.CameraInfo info = new Camera.CameraInfo();
Camera.getCameraInfo(mCameraId, info);
int rotation = activity.getWindowManager().getDefaultDisplay().getRotation();
int degrees = 0;
switch (rotation) {
case Surface.ROTATION_0:
degrees = 0;
break;
case Surface.ROTATION_90:
degrees = 90;
break;
case Surface.ROTATION_180:
degrees = 180;
break;
case Surface.ROTATION_270:
degrees = 270;
break;
}
int result;
if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
result = (info.orientation + degrees) % 360;
result = (360 - result) % 360; // compensate the mirror
} else { // back-facing
result = (info.orientation - degrees + 360) % 360;
}
LogUtil.i("The Angle at which the camera is rotated;" + result);
mOrienta = result;// This value has other uses
mCamera.setDisplayOrientation(result);
}
Copy the code
For aesthetics, make a full-screen theme ~
<! - the full screen - >
<style name="FullScreenTheme" parent="Theme.AppCompat.Light.NoActionBar">
<item name="android:windowFullscreen">true</item>
<item name="android:windowNoTitle">true</item>
</style>
Copy the code
Finally, remember to release the camera resources after using it.
if (null! = camera) { camera.stopPreview(); camera.release(); camera =null;
}
Copy the code
Project address: Github welcomes Start and fork