环球微动态丨Camera2API的基本功能笔记 Android实现Camera2预览和拍照效果
简介
【资料图】
网上对于 Camera2 的介绍有很多,在 Github 上也有很多关于 Camera2 的封装库,但是对于那些库,封装性太强,有时候我们仅仅是需要个简简单单的拍照功能而已,因此,自定义一个 Camera 使之变得轻量级那是非常重要的了。(本文并非重复造轮子, 而是在于学习 Camera2API 的基本功能, 笔记之。)
学习要点:
使用 Android Camera2 API 的基本功能。
迭代连接到设备的所有相机的特征。
显示相机预览和拍摄照片。
Camera2 API 为连接到 Android 设备的各个相机设备提供了一个界面。 它替代了已弃用的 Camera 类。
使用 getCameraIdList 获取所有可用摄像机的列表。 然后,您可以使用 getCameraCharacteristics,并找到适合您需要的最佳相机(前 / 后面,分辨率等)。
创建一个 CameraDevice.StateCallback 的实例并打开相机。 当相机打开时,准备开始相机预览。
使用 TextureView 显示相机预览。 创建一个 CameraCaptureSession 并设置一个重复的 CaptureRequest。
静像拍摄需要几个步骤。 首先,需要通过更新相机预览的 CaptureRequest 来锁定相机的焦点。
然后,以类似的方式,需要运行一个预捕获序列。之后,它准备拍摄一张照片。 创建一个新的 CaptureRequest 并调用 [capture] 。
完成后,别忘了解锁焦点。
实现效果环境
SDK>21
Camera2 类图
代码实现
CameraPreview.java
/**
* Created by shenhua on 2017-10-20-0020.
* Email shenhuanet@126.com
*/
public class CameraPreview extends TextureView {
private static final String TAG = "CameraPreview";
private static final SparseIntArray ORIENTATIONS = new SparseIntArray();//从屏幕旋转转换为JPEG方向
private static final int MAX_PREVIEW_WIDTH = 1920;//Camera2 API 保证的最大预览宽高
private static final int MAX_PREVIEW_HEIGHT = 1080;
private static final int STATE_PREVIEW = 0;//显示相机预览
private static final int STATE_WAITING_LOCK = 1;//焦点锁定中
private static final int STATE_WAITING_PRE_CAPTURE = 2;//拍照中
private static final int STATE_WAITING_NON_PRE_CAPTURE = 3;//其它状态
private static final int STATE_PICTURE_TAKEN = 4;//拍照完毕
private int mState = STATE_PREVIEW;
private int mRatioWidth = 0, mRatioHeight = 0;
private int mSensorOrientation;
private boolean mFlashSupported;
private Semaphore mCameraOpenCloseLock = new Semaphore(1);//使用信号量 Semaphore 进行多线程任务调度
private Activity activity;
private File mFile;
private HandlerThread mBackgroundThread;
private Handler mBackgroundHandler;
private Size mPreviewSize;
private String mCameraId;
private CameraDevice mCameraDevice;
private CaptureRequest.Builder mPreviewRequestBuilder;
private CaptureRequest mPreviewRequest;
private CameraCaptureSession mCaptureSession;
private ImageReader mImageReader;
static {
ORIENTATIONS.append(Surface.ROTATION_0, 90);
ORIENTATIONS.append(Surface.ROTATION_90, 0);
ORIENTATIONS.append(Surface.ROTATION_180, 270);
ORIENTATIONS.append(Surface.ROTATION_270, 180);
}
public CameraPreview(Context context) {
this(context, null);
}
public CameraPreview(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public CameraPreview(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
mFile = new File(getContext().getExternalFilesDir(null), "pic.jpg");
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
int width = MeasureSpec.getSize(widthMeasureSpec);
int height = MeasureSpec.getSize(heightMeasureSpec);
if (0 == mRatioWidth || 0 == mRatioHeight) {
setMeasuredDimension(width, height);
} else {
if (width < height * mRatioWidth / mRatioHeight) {
setMeasuredDimension(width, width * mRatioHeight / mRatioWidth);
} else {
setMeasuredDimension(height * mRatioWidth / mRatioHeight, height);
}
}
}
public void onResume(Activity activity) {
this.activity = activity;
startBackgroundThread();
//当Activity或Fragment OnResume()时,可以冲洗打开一个相机并开始预览,否则,这个Surface已经准备就绪
if (this.isAvailable()) {
openCamera(this.getWidth(), this.getHeight());
} else {
this.setSurfaceTextureListener(mSurfaceTextureListener);
}
}
public void onPause() {
closeCamera();
stopBackgroundThread();
}
public void setOutPutDir(File file) {
this.mFile = file;
}
public void setAspectRatio(int width, int height) {
if (width < 0 || height < 0) {
throw new IllegalArgumentException("Size can"t be negative");
}
mRatioWidth = width;
mRatioHeight = height;
requestLayout();
}
public void setAutoFlash(CaptureRequest.Builder requestBuilder) {
if (mFlashSupported) {
requestBuilder.set(CaptureRequest.CONTROL_AE_MODE,
CaptureRequest.CONTROL_AE_MODE_ON_AUTO_FLASH);
}
}
public void takePicture() {
lockFocus();
}
private void startBackgroundThread() {
mBackgroundThread = new HandlerThread("CameraBackground");
mBackgroundThread.start();
mBackgroundHandler = new Handler(mBackgroundThread.getLooper());
}
private void stopBackgroundThread() {
mBackgroundThread.quitSafely();
try {
mBackgroundThread.join();
mBackgroundThread = null;
mBackgroundHandler = null;
} catch (InterruptedException e) {
e.printStackTrace();
}
}
/**
* 处理生命周期内的回调事件
*/
private final TextureView.SurfaceTextureListener mSurfaceTextureListener = new TextureView.SurfaceTextureListener() {
@Override
public void onSurfaceTextureAvailable(SurfaceTexture texture, int width, int height) {
openCamera(width, height);
}
@Override
public void onSurfaceTextureSizeChanged(SurfaceTexture texture, int width, int height) {
configureTransform(width, height);
}
@Override
public boolean onSurfaceTextureDestroyed(SurfaceTexture texture) {
return true;
}
@Override
public void onSurfaceTextureUpdated(SurfaceTexture texture) {
}
};
/**
* 相机状态改变回调
*/
private final CameraDevice.StateCallback mStateCallback = new CameraDevice.StateCallback() {
@Override
public void onOpened(@NonNull CameraDevice cameraDevice) {
mCameraOpenCloseLock.release();
Log.d(TAG, "相机已打开");
mCameraDevice = cameraDevice;
createCameraPreviewSession();
}
@Override
public void onDisconnected(@NonNull CameraDevice cameraDevice) {
mCameraOpenCloseLock.release();
cameraDevice.close();
mCameraDevice = null;
}
@Override
public void onError(@NonNull CameraDevice cameraDevice, int error) {
mCameraOpenCloseLock.release();
cameraDevice.close();
mCameraDevice = null;
if (null != activity) {
activity.finish();
}
}
};
/**
* 处理与照片捕获相关的事件
*/
private CameraCaptureSession.CaptureCallback mCaptureCallback = new CameraCaptureSession.CaptureCallback() {
private void process(CaptureResult result) {
switch (mState) {
case STATE_PREVIEW: {
break;
}
case STATE_WAITING_LOCK: {
Integer afState = result.get(CaptureResult.CONTROL_AF_STATE);
if (afState == null) {
captureStillPicture();
} else if (CaptureResult.CONTROL_AF_STATE_FOCUSED_LOCKED == afState ||
CaptureResult.CONTROL_AF_STATE_NOT_FOCUSED_LOCKED == afState) {
Integer aeState = result.get(CaptureResult.CONTROL_AE_STATE);
if (aeState == null || aeState == CaptureResult.CONTROL_AE_STATE_CONVERGED) {
mState = STATE_PICTURE_TAKEN;
captureStillPicture();
} else {
runPreCaptureSequence();
}
}
break;
}
case STATE_WAITING_PRE_CAPTURE: {
Integer aeState = result.get(CaptureResult.CONTROL_AE_STATE);
if (aeState == null ||
aeState == CaptureResult.CONTROL_AE_STATE_PRECAPTURE ||
aeState == CaptureRequest.CONTROL_AE_STATE_FLASH_REQUIRED) {
mState = STATE_WAITING_NON_PRE_CAPTURE;
}
break;
}
case STATE_WAITING_NON_PRE_CAPTURE: {
Integer aeState = result.get(CaptureResult.CONTROL_AE_STATE);
if (aeState == null || aeState != CaptureResult.CONTROL_AE_STATE_PRECAPTURE) {
mState = STATE_PICTURE_TAKEN;
captureStillPicture();
}
break;
}
}
}
@Override
public void onCaptureProgressed(@NonNull CameraCaptureSession session,
@NonNull CaptureRequest request,
@NonNull CaptureResult partialResult) {
process(partialResult);
}
@Override
public void onCaptureCompleted(@NonNull CameraCaptureSession session,
@NonNull CaptureRequest request,
@NonNull TotalCaptureResult result) {
process(result);
}
};
/**
* 在确定相机预览大小后应调用此方法
*
* @param viewWidth 宽
* @param viewHeight 高
*/
private void configureTransform(int viewWidth, int viewHeight) {
if (null == mPreviewSize || null == activity) {
return;
}
int rotation = activity.getWindowManager().getDefaultDisplay().getRotation();
Matrix matrix = new Matrix();
RectF viewRect = new RectF(0, 0, viewWidth, viewHeight);
RectF bufferRect = new RectF(0, 0, mPreviewSize.getHeight(), mPreviewSize.getWidth());
float centerX = viewRect.centerX();
float centerY = viewRect.centerY();
if (Surface.ROTATION_90 == rotation || Surface.ROTATION_270 == rotation) {
bufferRect.offset(centerX - bufferRect.centerX(), centerY - bufferRect.centerY());
matrix.setRectToRect(viewRect, bufferRect, Matrix.ScaleToFit.FILL);
float scale = Math.max(
(float) viewHeight / mPreviewSize.getHeight(),
(float) viewWidth / mPreviewSize.getWidth());
matrix.postScale(scale, scale, centerX, centerY);
matrix.postRotate(90 * (rotation - 2), centerX, centerY);
} else if (Surface.ROTATION_180 == rotation) {
matrix.postRotate(180, centerX, centerY);
}
this.setTransform(matrix);
}
/**
* 根据mCameraId打开相机
*/
private void openCamera(int width, int height) {
setUpCameraOutputs(width, height);
configureTransform(width, height);
CameraManager manager = (CameraManager) getContext().getSystemService(Context.CAMERA_SERVICE);
try {
if (!mCameraOpenCloseLock.tryAcquire(2500, TimeUnit.MILLISECONDS)) {
throw new RuntimeException("Time out waiting to lock camera opening.");
}
if (ActivityCompat.checkSelfPermission(activity, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {
// TODO: Consider calling
// ActivityCompat#requestPermissions
// here to request the missing permissions, and then overriding
// public void onRequestPermissionsResult(int requestCode, String[] permissions,
// int[] grantResults)
// to handle the case where the user grants the permission. See the documentation
// for ActivityCompat#requestPermissions for more details.
return;
}
manager.openCamera(mCameraId, mStateCallback, mBackgroundHandler);
} catch (CameraAccessException e) {
e.printStackTrace();
} catch (InterruptedException e) {
throw new RuntimeException("Interrupted while trying to lock camera opening.", e);
}
}
/**
* 关闭相机
*/
private void closeCamera() {
try {
mCameraOpenCloseLock.acquire();
if (null != mCaptureSession) {
mCaptureSession.close();
mCaptureSession = null;
}
if (null != mCameraDevice) {
mCameraDevice.close();
mCameraDevice = null;
}
if (null != mImageReader) {
mImageReader.close();
mImageReader = null;
}
} catch (InterruptedException e) {
throw new RuntimeException("Interrupted while trying to lock camera closing.", e);
} finally {
mCameraOpenCloseLock.release();
}
}
/**
* 设置相机相关的属性或变量
*
* @param width 相机预览的可用尺寸的宽度
* @param height 相机预览的可用尺寸的高度
*/
@SuppressWarnings("SuspiciousNameCombination")
private void setUpCameraOutputs(int width, int height) {
CameraManager manager = (CameraManager) getContext().getSystemService(Context.CAMERA_SERVICE);
try {
for (String cameraId : manager.getCameraIdList()) {
CameraCharacteristics characteristics = manager.getCameraCharacteristics(cameraId);
// 在这个例子中不使用前置摄像头
Integer facing = characteristics.get(CameraCharacteristics.LENS_FACING);
if (facing != null && facing == CameraCharacteristics.LENS_FACING_FRONT) {
continue;
}
StreamConfigurationMap map = characteristics.get(CameraCharacteristics.SCALER_STREAM_CONFIGURATION_MAP);
if (map == null) {
continue;
}
Size largest = Collections.max(Arrays.asList(map.getOutputSizes(ImageFormat.JPEG)),
new CompareSizesByArea());
mImageReader = ImageReader.newInstance(largest.getWidth(), largest.getHeight(),
ImageFormat.JPEG, /*maxImages*/2);
mImageReader.setOnImageAvailableListener(
mOnImageAvailableListener, mBackgroundHandler);
int displayRotation = activity.getWindowManager().getDefaultDisplay().getRotation();
// noinspection ConstantConditions
mSensorOrientation = characteristics.get(CameraCharacteristics.SENSOR_ORIENTATION);
boolean swappedDimensions = false;
switch (displayRotation) {
case Surface.ROTATION_0:
case Surface.ROTATION_180:
if (mSensorOrientation == 90 || mSensorOrientation == 270) {
swappedDimensions = true;
}
break;
case Surface.ROTATION_90:
case Surface.ROTATION_270:
if (mSensorOrientation == 0 || mSensorOrientation == 180) {
swappedDimensions = true;
}
break;
default:
Log.e(TAG, "Display rotation is invalid: " + displayRotation);
}
Point displaySize = new Point();
activity.getWindowManager().getDefaultDisplay().getSize(displaySize);
int rotatedPreviewWidth = width;
int rotatedPreviewHeight = height;
int maxPreviewWidth = displaySize.x;
int maxPreviewHeight = displaySize.y;
if (swappedDimensions) {
rotatedPreviewWidth = height;
rotatedPreviewHeight = width;
maxPreviewWidth = displaySize.y;
maxPreviewHeight = displaySize.x;
}
if (maxPreviewWidth > MAX_PREVIEW_WIDTH) {
maxPreviewWidth = MAX_PREVIEW_WIDTH;
}
if (maxPreviewHeight > MAX_PREVIEW_HEIGHT) {
maxPreviewHeight = MAX_PREVIEW_HEIGHT;
}
mPreviewSize = chooseOptimalSize(map.getOutputSizes(SurfaceTexture.class),
rotatedPreviewWidth, rotatedPreviewHeight, maxPreviewWidth,
maxPreviewHeight, largest);
int orientation = getResources().getConfiguration().orientation;
if (orientation == Configuration.ORIENTATION_LANDSCAPE) {
setAspectRatio(mPreviewSize.getWidth(), mPreviewSize.getHeight());
} else {
setAspectRatio(mPreviewSize.getHeight(), mPreviewSize.getWidth());
}
Boolean available = characteristics.get(CameraCharacteristics.FLASH_INFO_AVAILABLE);
mFlashSupported = available == null ? false : available;
mCameraId = cameraId;
return;
}
} catch (CameraAccessException e) {
e.printStackTrace();
} catch (NullPointerException e) {
Log.e(TAG, "设备不支持Camera2");
}
}
/**
* 获取一个合适的相机预览尺寸
*
* @param choices 支持的预览尺寸列表
* @param textureViewWidth 相对宽度
* @param textureViewHeight 相对高度
* @param maxWidth 可以选择的最大宽度
* @param maxHeight 可以选择的最大高度
* @param aspectRatio 宽高比
* @return 最佳预览尺寸
*/
private static Size chooseOptimalSize(Size[] choices, int textureViewWidth, int textureViewHeight,
int maxWidth, int maxHeight, Size aspectRatio) {
List bigEnough = new ArrayList<>();
List notBigEnough = new ArrayList<>();
int w = aspectRatio.getWidth();
int h = aspectRatio.getHeight();
for (Size option : choices) {
if (option.getWidth() <= maxWidth && option.getHeight() <= maxHeight &&
option.getHeight() == option.getWidth() * h / w) {
if (option.getWidth() >= textureViewWidth &&
option.getHeight() >= textureViewHeight) {
bigEnough.add(option);
} else {
notBigEnough.add(option);
}
}
}
if (bigEnough.size() > 0) {
return Collections.min(bigEnough, new CompareSizesByArea());
} else if (notBigEnough.size() > 0) {
return Collections.max(notBigEnough, new CompareSizesByArea());
} else {
Log.e(TAG, "Couldn"t find any suitable preview size");
return choices[0];
}
}
/**
* 为相机预览创建新的CameraCaptureSession
*/
private void createCameraPreviewSession() {
try {
SurfaceTexture texture = this.getSurfaceTexture();
assert texture != null;
// 将默认缓冲区的大小配置为想要的相机预览的大小
texture.setDefaultBufferSize(mPreviewSize.getWidth(), mPreviewSize.getHeight());
Surface surface = new Surface(texture);
mPreviewRequestBuilder = mCameraDevice.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW);
mPreviewRequestBuilder.addTarget(surface);
// 我们创建一个 CameraCaptureSession 来进行相机预览
mCameraDevice.createCaptureSession(Arrays.asList(surface, mImageReader.getSurface()),
new CameraCaptureSession.StateCallback() {
@Override
public void onConfigured(@NonNull CameraCaptureSession cameraCaptureSession) {
if (null == mCameraDevice) {
return;
}
// 会话准备好后,我们开始显示预览
mCaptureSession = cameraCaptureSession;
try {
mPreviewRequestBuilder.set(CaptureRequest.CONTROL_AF_MODE,
CaptureRequest.CONTROL_AF_MODE_CONTINUOUS_PICTURE);
setAutoFlash(mPreviewRequestBuilder);
mPreviewRequest = mPreviewRequestBuilder.build();
mCaptureSession.setRepeatingRequest(mPreviewRequest, mCaptureCallback, mBackgroundHandler);
} catch (CameraAccessException e) {
e.printStackTrace();
}
}
@Override
public void onConfigureFailed(@NonNull CameraCaptureSession cameraCaptureSession) {
}
}, null);
} catch (CameraAccessException e) {
e.printStackTrace();
}
}
/**
* 从指定的屏幕旋转中检索照片方向
*
* @param rotation 屏幕方向
* @return 照片方向(0,90,270,360)
*/
private int getOrientation(int rotation) {
return (ORIENTATIONS.get(rotation) + mSensorOrientation + 270) % 360;
}
/**
* 锁定焦点
*/
private void lockFocus() {
try {
// 如何通知相机锁定焦点
mPreviewRequestBuilder.set(CaptureRequest.CONTROL_AF_TRIGGER, CameraMetadata.CONTROL_AF_TRIGGER_START);
// 通知mCaptureCallback等待锁定
mState = STATE_WAITING_LOCK;
mCaptureSession.capture(mPreviewRequestBuilder.build(), mCaptureCallback, mBackgroundHandler);
} catch (CameraAccessException e) {
e.printStackTrace();
}
}
/**
* 解锁焦点
*/
private void unlockFocus() {
try {
mPreviewRequestBuilder.set(CaptureRequest.CONTROL_AF_TRIGGER,
CameraMetadata.CONTROL_AF_TRIGGER_CANCEL);
setAutoFlash(mPreviewRequestBuilder);
mCaptureSession.capture(mPreviewRequestBuilder.build(), mCaptureCallback,
mBackgroundHandler);
mState = STATE_PREVIEW;
mCaptureSession.setRepeatingRequest(mPreviewRequest, mCaptureCallback,
mBackgroundHandler);
} catch (CameraAccessException e) {
e.printStackTrace();
}
}
/**
* 拍摄静态图片
*/
private void captureStillPicture() {
try {
if (null == activity || null == mCameraDevice) {
return;
}
final CaptureRequest.Builder captureBuilder =
mCameraDevice.createCaptureRequest(CameraDevice.TEMPLATE_STILL_CAPTURE);
captureBuilder.addTarget(mImageReader.getSurface());
captureBuilder.set(CaptureRequest.CONTROL_AF_MODE,
CaptureRequest.CONTROL_AF_MODE_CONTINUOUS_PICTURE);
setAutoFlash(captureBuilder);
// 方向
int rotation = activity.getWindowManager().getDefaultDisplay().getRotation();
captureBuilder.set(CaptureRequest.JPEG_ORIENTATION, getOrientation(rotation));
CameraCaptureSession.CaptureCallback captureCallback
= new CameraCaptureSession.CaptureCallback() {
@Override
public void onCaptureCompleted(@NonNull CameraCaptureSession session,
@NonNull CaptureRequest request,
@NonNull TotalCaptureResult result) {
Toast.makeText(getContext(), "Saved: " + mFile, Toast.LENGTH_SHORT).show();
Log.d(TAG, mFile.toString());
unlockFocus();
}
};
mCaptureSession.stopRepeating();
mCaptureSession.abortCaptures();
mCaptureSession.capture(captureBuilder.build(), captureCallback, null);
} catch (CameraAccessException e) {
e.printStackTrace();
}
}
/**
* 运行preCapture序列来捕获静止图像
*/
private void runPreCaptureSequence() {
try {
// 设置拍照参数请求
mPreviewRequestBuilder.set(CaptureRequest.CONTROL_AE_PRECAPTURE_TRIGGER,
CaptureRequest.CONTROL_AE_PRECAPTURE_TRIGGER_START);
mState = STATE_WAITING_PRE_CAPTURE;
mCaptureSession.capture(mPreviewRequestBuilder.build(), mCaptureCallback, mBackgroundHandler);
} catch (CameraAccessException e) {
e.printStackTrace();
}
}
/**
* 比较两者大小
*/
private static class CompareSizesByArea implements Comparator {
@Override
public int compare(Size lhs, Size rhs) {
return Long.signum((long) lhs.getWidth() * lhs.getHeight() -
(long) rhs.getWidth() * rhs.getHeight());
}
}
/**
* ImageReader的回调对象
*/
private final ImageReader.OnImageAvailableListener mOnImageAvailableListener
= new ImageReader.OnImageAvailableListener() {
@Override
public void onImageAvailable(ImageReader reader) {
mBackgroundHandler.post(new ImageSaver(reader.acquireNextImage(), mFile));
}
};
/**
* 将捕获到的图像保存到指定的文件中
*/
private static class ImageSaver implements Runnable {
private final Image mImage;
private final File mFile;
ImageSaver(Image image, File file) {
mImage = image;
mFile = file;
}
@Override
public void run() {
ByteBuffer buffer = mImage.getPlanes()[0].getBuffer();
byte[] bytes = new byte[buffer.remaining()];
buffer.get(bytes);
FileOutputStream output = null;
try {
output = new FileOutputStream(mFile);
output.write(bytes);
} catch (IOException e) {
e.printStackTrace();
} finally {
mImage.close();
if (null != output) {
try {
output.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
}
}
MainActivity.java
public class MainActivity extends AppCompatActivity {
CameraPreview cameraView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
cameraView = (CameraPreview) findViewById(R.id.cameraView);
}
@Override
protected void onResume() {
super.onResume();
cameraView.onResume(this);
}
@Override
protected void onPause() {
cameraView.onPause();
super.onPause();
}
public void takePic(View view) {
cameraView.takePicture();
}
}
activity_main.xml
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/constraintLayout"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@android:color/black"
tools:context="com.shenhua.ocr.activity.Main2Activity">
android:id="@+id/cameraView"
android:layout_width="wrap_content"
android:layout_height="wrap_content" />
android:layout_width="70dp"
android:layout_height="70dp"
android:layout_gravity="center"
android:background="@drawable/ic_capture_200px"
android:onClick="takePic"
android:text="TAKE"
app:layout_constraintBottom_toBottomOf="@id/constraintLayout"
app:layout_constraintEnd_toEndOf="@id/constraintLayout"
app:layout_constraintStart_toStartOf="@id/constraintLayout"
app:layout_constraintTop_toTopOf="@id/cameraView"
app:layout_constraintVertical_bias="0.97" />
资源文件 ic_capture_200px.xml
android:viewportWidth="1024.0" android:width="24dp" xmlns:android="http://schemas.android.com/apk/res/android">
其它
Manifest 权限:
Android6.0 运行时权限未贴出。(注意:为了方便读者手机端阅读,本文代码部分的成员变量使用了行尾注释,在正常编程习惯中,请使用 /* / 注释。)
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。
标签:
相关推荐:
精彩放送:
- []世界视讯!如何搭建个人网站?创建网站的第一步是什么?
- []企业安全生产管理制度包括哪些?安全操作规程
- []今日播报!applet是什么文件夹可以删除吗?applet文件夹详情介绍
- []2022兰州交通大学研究生招生简章发布 兰州交通大学研究生
- []68岁成龙探班杨幂开心比耶,杨幂新电影造型曝光,二人交情超20年
- []每日焦点!小学一年级的目的组词有哪些?目的怎么组词?
- []全球观天下!上古卷轴5怎么结婚?结婚流程是怎样的?
- []全球微资讯!abb变频电机怎么样?abb变频电机价格是多少?
- []焦点日报:百度闪付如何添加银行卡?百度闪付添加银行卡方法
- []联想笔记本一键恢复7.0怎么用?详细方法步骤
- []天天观速讯丨华为nova5手机无法开机怎么办?华为nova5手机维修
- []C.Aurum潮金:黄金珠宝行业内的新锐时尚品牌
- []今日快看!东航随心飞怎么买?东方万里行会员怎么绑定?
- []当前快看:例如的英语是什么?例如的英文说法
- []A股酒店及餐饮板块持续拉升 同庆楼、君亭酒店等均涨超6%
- []天天要闻:黑洞灰洞有什么区别?黑洞灰洞详情介绍
- []世界动态:重庆铁山坪位置在哪?关于重庆铁山坪的介绍
- []vr场景怎么做?VR视频制作教程
- []世界要闻:德国曼瑞德温控器怎么样?德国曼瑞德温控器介绍
- []天天速看:唐太宗李世民总共有几个儿子?李世民十四个儿子的结局如何?
- []金松冰箱怎么样?金松冰箱多少钱?详细介绍
- []广东南海控股10亿元公司债利率不调整 仍为3.02%
- []关注:【数模竞赛】Algorithm:数学建模大赛(CUMCM)
- []天天滚动:课外常识:赋闲的意思是什么?赋闲一词详情介绍
- []全球简讯:中国电池企业前十名有哪些?中国电池企业前十名介绍
- []天天消息!扫雷和空当接龙freecell:算法深度优先
- []世界速讯:MS-TTS:免费微软TTS语音合成工具 一键合成导出MP3音频
- []世界快消息!幽魂碎片是什么?幽魂碎片在哪儿换?
- []环球实时:张一鸣:华为人才基因的真正密码
- []当前播报:下载软件哪个好?迅雷、迅雷和比特彗星下载工具介绍
- []世界微头条丨nokia5300怎么刷机?nokia5300刷机教程及手机优势
- []Word文档中如何输入汉字的偏旁部首?输入汉字的偏旁部首方法
- []每日播报!ios8越狱后必装插件有哪些?ios8越狱后必装插件汇总
- []每日速看!如何注册网站域名?注册网站域名需要注意哪些?
- []世界时讯:关于金鹰独播剧场电视机列表 你了解多少?
- []环球即时看!肃穆的意思是什么?肃穆的意思大全
- []【世界独家】宿建德江的诗意 关于宿建德江的诗意20字
- []最低配置是什么意思?赛博朋克2077配置要求2022
- []世界速读:最新手机壳价格怎样?最新手机壳报价大全
- []世界快播:你的qq号几年了?回顾腾讯QQ的成长发展史(1998-2016年)
- []d3dx9_34.dll是什么?系统d3dx9_34.dll错误会带来什么危害?
- []视频会议系统是什么意思?有哪些功能?
- []马冬晗学习计划表 清华学霸计划表曝光
- []世界微资讯!福州连江:延长人才就业购房补助期限至6月30日
- []新动态:微信塞班版为什么退出后会自动启动?问题出在哪里?
- []“公司一天成交500多套房!”北京中介:忙得没时间吃饭!京沪二手房挂牌价上涨,深圳也在回暖
- []摩托罗拉xt532怎么刷机?摩托罗拉xt532刷机教程介绍
- []天天信息:【技术】硬盘存储器的层次结构及原理
- []动态焦点:RMSE、MAE、MSE 如何衡量模型效果好坏?
- []【天天新视野】华昌化工:纯碱年产量为66万吨
- []环球微动态丨瑞士信贷:上调百胜中国目标价 预期其今明两年同店销售增长13%及3%
- []【当前热闻】金融市场的功能有哪些?金融市场的功能介绍
- []环球微速讯:高效空气过滤器有什么作用?高效空气过滤器作用介绍
- []每日视点!如何在Win10登陆界面添加签名?Win10登陆界面添加签名方法
- []焦点日报:如何让视障用户更好使用你的网站?10条网站易用性技巧
- []焦点快播:奥斯特电流实验flash制作 关于奥斯特实验的那些故事
- []bose家庭影院好吗?bose家庭影院有哪些优势?
- []【反汇编】ce附加红警3找钱的进程
- []【全球聚看点】世界黄金协会:黄金ETF第九个月资金流出,但金价跌势正在逆转!
- []【世界播资讯】供应商的选择、评审和动态管理的方法
- []9个学习资源分享给大家 推荐几个资源聚集地
- []焦点热门:除和除以以及被除的区别是什么?详情介绍
- []“橡皮擦”用英语该怎么说?橡皮擦的英文说法
- []全球今日讯!【十大排序算法】十大排序算法总结
- []焦点简讯:短网址移动网缩短接口网址api调用php方式解析
- []当前热讯:王长贵怎么死的?王长贵是一个什么样的角色?
- []【焦点热闻】东营有什么好玩的地方?东营旅游景点有哪些?
- []鸵鸟生活在什么地方?鸵鸟的生活习性是怎样的?
- []【新要闻】新版手机qq怎么换皮肤?新版手机qq换皮肤的方法?
- []今日看点:c罗拿过几次金球奖?c罗拿过的奖项有哪些?
- []【天天报资讯】广西有哪些乐队?广西乐队汇总?
- []天天热点!书名带邪字的小说有哪些?书名带邪字的小说汇总?
- []悠悠寸草心这句话是什么意思?悠悠寸草心出处是哪里?
- []当前视点!2~3人吃蛋糕几寸合适?蛋糕尺寸怎么选择?
- []如何用手机定位防盗?用手机定位防盗的方法步骤?
- []天天视讯!信用卡微信快捷支付怎么开通(信用卡未开通快捷支付怎么开通)
- []八大券商主题策略:关注一号文件和转基因政策落地进程!养殖股的机会来了?
- []每日热文:香港1个月HIBOR连跌4日
- []天天播报:【BT金融分析师】甲骨文股价暴涨近50%,分析师称其收入已停滞10年
- []环球关注:成都天府新区投资集团50亿小公募获深交所通过
- []香港荃湾宝丰路住宅官地至少收10份标书
- []【环球快播报】博瑞传播:我司持有成都每经传媒有限公司35%股份,为公司参股公司
- []海昌海洋公园股价跌13.51% 或因被剔除出MSCI中国指数
- []今头条!穿刺是什么
- []全球讯息:港2022年全年逾2.1万伙私宅落成创十八年新高
- []天天观速讯丨中科海讯:公司目前收入主要来源于批生产项目,部分收入来源于研发项目
- []湖北南漳抽水蓄能电站项目获核准
- []世界百事通!埃泰斯重点支持第十三届中国国际储能大会召开
- []建设银行 公积金消费贷(建设银行公积金消费贷款叫什么)
- []全球视点!城市民宿应与社区“共生”
- []全球消息!四川力破缺电之困 布局储能或解短时困难
- []热消息:预售35万元起 智己LS7今日正式上市 科技感拉满速速入手
- []环球热头条丨上海旅游市场恢复进程中可能出现的问题及应对措施
- []世界速看:湖北能源集团南漳张家坪抽水蓄能电站签约
- []天天快讯:国信期货早评:铁矿石偏强震荡,铁合金短期反弹,玻璃弱势
- []中建三局50亿元超短债完成发行 期限42日
- []世界快看:聆达股份:赛拉弗是公司电池片业务客户。公司与客户们向来都寻求或保持着紧密的合作交流
- []当前焦点!农业银行社保卡初始密码,123456和111111
- []世界微速讯:宁波精达:我司会按照相关减持规定履行披露义务,请关注我司公告
- []天天滚动:双环科技:公司目前纯碱库存很低
- 【天天新视野】美国“至暗时刻”尚未过去!又一大佬警告:通胀料持续高企甚至反弹
- 全球观速讯丨CEO欲减持42亿,迈瑞医疗还值得期待吗?
- 当前短讯!兴业银行卡丢失了怎么办(兴业银行卡丢了怎么补)
- 前沿热点:采日能源拿下200MWh新订单
- 民生信用卡永久免年费吗(民生信用卡免年费的优惠)
- 世界今热点:“锂钠比翼”,雄韬股份25GWh新能源电池项目签约荆门
- 【世界播资讯】143MW/286MWh!中广核电网侧储能电站项目EPC中标公示
- 最新发布!2022全球动力电池装机量TOP10
- 中国银行信用卡怎么一次性还清全款(中国银行信用卡怎么全额还款)
- 世界速递!智光电气:目前储能公司已申请专利88项;储能公司已获得授权专利52项
- 【世界快播报】标普确认龙湖集团投资级评级,展望上调为稳定
- 全球观天下!镇江城建1.6亿元超短期融资券将于2月19日兑付
- 三元生物:该项目为生物质供热项目,不进行热电联产
- 快报:安徽省土地推介超130宗 滁州市区8宗涉宅地出让面积约982亩
- 环球百事通!彭州城投15亿元私募债项目状态更新为“已受理”
- 贵阳贵安新区:启动建设5000套保租房及4万户老旧小区改造
- 环球今头条!飞龙股份:公司的电子水泵系列产品和热管理系统产品可以应用在汽车和充电桩领域
- 【天天报资讯】学校慰问退休老师送什么礼物好
- 超40个重大项目年内开工 贵州新能源产业跑出加速度
- 环球短讯!如何成功打造一个酒店品牌?
- 世界信息:投保人是父母(请输入投保人手机号)
- 每日快看:起底头部酒店集团的生活方式酒店版图
- 【世界播资讯】百川股份:子公司海基新能源新发布的钠离子电池,可应用于低速车和两轮车
- 微动态丨大名城扭亏为盈AB面:抓紧募资“补血”,旗下楼盘无证施工被罚
- 嘉益股份:公司若有相关事宜,会按照相关法律法规履行信息披露义务,请以公司公告为准
- 全球关注:大悦城收深交所关注函:要求说明2022年业绩由盈转亏且亏损集中在第四季度的原因
- 全国百城1月新房成交规模创近8年新低,深圳等15城新房成交量以环比4倍增幅迅速回暖
- 当前热点-508.5亿!1月房企融资规模降幅收窄,金茂等多家房企境外发债“开门红”
- 世界头条:北元集团:公司从未与任何股东私下达成任何形式的协议,不存在应披露而未披露的重大事项
- 世界资讯:全国300城1月卖地收入同比降超4成,绿城87亿领跑房企新增货值榜
- 热推荐:北京利尔:公司目前拥有菱镁矿资源一处,未来将根据公司发展需要,审慎关注矿产资源投资机会祝您投资顺利
- 【全球聚看点】如何贷款7万(怎么贷款7万)
- 今日关注:中国人寿的交强险多少钱(交强险一般中国人寿多少钱一年多少)
- 全球快资讯丨旅游订单环比增长121%,《狂飙》背后的城市准备好了吗?
- 要闻:华泰证券:2024年或成钙钛矿电池量产元年
- 环球观点:基金积极布局“景气”产品
- 天天头条:Expedia Q4住宿业务收入创新高,营销ROI未见改善
- 反弹行情中业绩首尾相差近70% 这些主动权益基金率先“回血”
- 当前头条:中金公司:恒指调整要关注部分新经济公司纳入的潜在可能性
- 实时:酒店数字营销的底层路径——营销文本的生成案例
- 每日消息!句子大全霸气古风短句(优选221句)
- 微资讯!石家庄城投公司28.33亿元摘得石家庄主城区6宗宅地
- 每日报道:金华东阳成功出让4宗小体量相邻地块 收金2570万元
- 原油交易提醒:土耳其地震对供应影响降低,油价回落,警惕地缘风险加剧
- 视焦点讯!2022年法拍房挂拍量创新高,但成交不足两成
- 当前热点-三达膜:公司在此领域的业务主要集中在核电站的高硅水处理
- 【当前热闻】万马股份:万马爱充充电设备电压目前已支持高压平台充电,最高可以支持到1000V
- 每日速读!越秀地产:附属公司广州城建申请发行本金最高为94亿元公司债
- 今日热文:凯乐科技将被强制退市 受损投资者可索赔
- 天天新资讯:龙父之牙任务详细全攻略
- 交通银行备用金怎么申请(交通银行备用金怎么申请)
- 上海社保转出后又回上海工作可以吗(上海社保转出后又回上海工作了)
- 深圳医保卡5000才能买药(深圳医保卡超过多少可以买药)
- 世界今日讯!刘亦菲带火的云南小镇,后来怎么样了
- 天天看点:结构改善,1月房企非银融资逾508亿元
- 世界观天下!卓越物业转让铂樾府物业65%股权予葵涌颐和物业
- 环球快报:佛山楼市全面取消限购两个月:成交额依旧同比腰斩,豪宅市场异军突起
- 【世界新视野】不降价卖不动 武汉将房地产归为“困难行业”
- 精选!TD早报 | 多家航司将停售低价机票?民航局回应:未提相关要求;迪士尼计划裁员7000人
- 外地社保转回来后怎么计算(外地社保转回来后怎么计算)
- 校园迎来三方面变化
- 全球速递!信用卡办的车贷可以提前还吗现在(信用卡分期24期可以提前还吗)
- 宋炯:施耐德电气直流领域的“绿色创新智慧”
- 环球热消息:四大行那个信用卡额度高(四大行信用卡哪个额度高)
- 社保基数可以高于工资吗(社保基数可以高于工资吗)
- 世界焦点!在网上申请信用卡会上征信吗(网上申请信用卡影响征信吗)
- 世界热推荐:生育险可以退么(给离职员工的生育保险可以退吗)
- 天天看点:中国平安税优识别码,在保单右上方
- 世界今亮点!美股异动 | 新东方(EDU.US)涨超6% 近日宣布参建自营品工厂
- 边境牧羊犬好养吗
- 阜阳新型冠状病毒肺炎疫情:2月10日阜阳疫情最新消息今天数据统计情况通报
- 全球微头条丨清明节的名言名句
- 社保信息被填写为农村户口怎么办(社保填农村户口还是城镇户口好)
- 环球快报:1985年,许世友最后一个生日的照片,此时已80岁高龄已是肝癌晚期
- 环球热文:新签储能产品销售框架协议金额超2021年全年收入 德宏股份提示多项风险
- 长缆科技:2月8日公司高管薛奇减持公司股份合计4.5万股
- 【天天快播报】中超控股:2月8日公司高管霍振平减持公司股份合计2.5万股
- 天禾股份:2月8日公司高管罗旋彬减持公司股份合计10万股
- 环球新资讯:民生卡被限制交易了,怎么处理(民生卡限额了怎么解除)
- 当前讯息:浦发银行还贷款怎么还款(浦发银行app怎么提前还贷)
- 世界信息:平安保单贷款如何还款(平安保单贷款还款方法)
- 天天播报:南通房贷可以商转公吗(南京商贷转公贷最新规定)
- 你租房花多少?报告:半数人认为占收入10%-20%最佳
- 新资讯:值得买:2月8日公司高管刘超减持公司股份合计1.47万股
- 当前速读:华瑞股份:2月8日公司高管孙瑞娣减持公司股份合计25万股
- 当前头条:新城吾悦广场首进杭州,新城25.37亿摘得杭州崇贤地块
- 天天即时:ST开元:2月8日公司高管江勇减持公司股份合计49万股
- 看点:周鸿祎谈ChatGPT:肯定有泡沫,但不是坏事!搭不上这班车容易被淘汰
- 全球新动态:医保卡买过的药能查到吗(医保卡买过的药能查到吗)
- 世界聚焦:超帐户授权仅允许或仅控制参数表限制(超账户授权仅允许或仅控制参数表限额信用卡分期)
- 【天天聚看点】陈文静:近两年重庆法拍房成交套数最多
- 环球今热点:58同城、安居客发布《2023节后返城租房调查报告》:85.7%的意向租房人群选择整租
- 每日消息!周陆啦个人资料
- 中公高科:2月6日公司高管李强减持公司股份合计4000股
- 【天天时快讯】新安股份:2月8日至2月9日公司高管姜永平减持公司股份合计13.5万股
- 广州市住建局:支持项目“拿地即开工”
- 招商蛇口1月签约销售金额157.67亿元 同比增加4.09%
- 快播:菱电电控:2月8日周良润减持公司股份合计2000股
- 热文:退休2年后死亡养老金能全退吗(退休后死亡领139个月工资)
- 世界观焦点:情人节送男朋友什么礼物呢,参考这份有趣的礼品清单