我有一个对话框设计,其中相机预览应该是纵向模式(请参阅图片)。我通过getOptimalPreviewSize()正确选择预览大小,当我不使用setDisplayOrientation(90)时,我得到了这个:
当我使用setDisplayOrientation(90)时,我得到了这个:
有人有任何想法如何解决它?安卓可以做那种事吗?
回答(感谢Eddy Talvala):
由于当xml中的凸轮视图为水平时,您无法纵向处理整个凸轮图像,因此应进行裁剪。我决定水平放置并裁掉底部的所有东西:
| |
| _________ | _____
|| || | |
|| || | ^-^ |
|| ______ || | \_/ |
|| | ^-^ ||| | | |
|| |_\_/_ ||| |__|__|
|| image || sensor
|| ||
|| ||
||_________||
|| < O [] ||
|___________|
device
您应该使用TextureView指定适合和裁剪的矩阵。像那样的东西(kotlin):
fun camInit() {
val camParam = camera.parameters
val optimalSize = getOptimalPortraitPreviewSize(camParam.supportedPreviewSizes, width, height)
camParam.setPreviewSize(optimalSize.width, optimalSize.height)
camera.parameters = camParam
camera.setPreviewTexture(surface)
val scaleDelta = optimalSize.height - preview.width // for portrait
val scaleY: Float = (optimalSize.width.toFloat() - scaleDelta) / preview.height // for portrait
val matrix = Matrix()
matrix.setScale(1f, scaleY) // 1f cause we fit horizontally
preview.setTransform(matrix)
camera.startPreview()
}
请注意,对于纵向模式,您应该使用特定的getOptimalProtraitPreviewSize(),导致每个人用来获取相机预览最佳尺寸的标准功能(使用ASPECT_TOLERANCE和其他东西)可以返回一个小分辨率的尺寸。尝试这样的事情:
fun getOptimalPortraitPreviewSize(sizes: List<Camera.Size>, previewWidth: Int) {
var selectedSize: Camera.Size = sizes[0]
for (size in sizes) {
// use size's height cause of portrait
val currentSizeWidthDelta = abs(size.height - previewWidth)
val selectedSizeWidthDelta = abs(selectedSize.height - previewWidth)
if (currentSizeWidthDelta < selectedSizeWidthDelta) selectedSize = size
}
return selectedSize
}
答案 0 :(得分:3)
图像传感器的长边与设备的长边对齐。
(bad ascii art):
____________
| |
| _________ | _____
|| || | |
|| || | |
|| || | |
|| || | |
|| || |_____|
|| || sensor
|| || physical
|| || orientation
||_________||
|| < O [] ||
|___________|
device
鉴于此,你无法像这样绘制图像:
_____________
| |
| _________ | _____
|| || | |
|| || | |
|| ______ || | |
|| | ||| | |
|| |______||| |_____|
|| image || sensor
|| ||
|| ||
||_________||
|| < O [] ||
|___________|
device
除非你要么旋转图像(在这种情况下不能向上):
_____________
| |
| _________ | _____
|| || | |
|| || | ^-^ |
|| ______ || | \_/ |
|| | <-\__||| | | |
|| |_<-/__||| |__|__|
|| image || sensor
|| ||
|| ||
||_________||
|| < O [] ||
|___________|
device
或者你水平拉伸图像,看起来很糟糕:
_____________
| |
| _________ | _____
|| || | |
|| || | ^-^ |
|| ______ || | \_/ |
|| | ^___^||| | | |
|| |___|__||| |__|__|
|| image || sensor
|| ||
|| ||
||_________||
|| < O [] ||
|___________|
device
或者裁剪图像的一部分,这会大大减少FOV:
_____________
| |
| _________ | _____
|| || | |
|| || | ^-^ |
|| ______ || | \_/ |
|| | ^-^ ||| | | |
|| |_\_/__||| |__|__|
|| image || sensor
|| ||
|| ||
||_________||
|| < O [] ||
|___________|
device
因为当手机是风景时图像传感器是横向的,所以如果没有发生这三种情况中的任何一种,就无法在纵向UI中放置横向预览。您需要在纵向UI中进行纵向预览,或者需要裁剪图像。这不是Android的限制,它只是几何的限制。
如果你想裁剪,你可能想要将相机数据发送到SurfaceTexture,然后在OpenGL中裁剪,如果你想要实时预览。
答案 1 :(得分:0)
如果您使用的是Camera2 API。当您通过ChooseOptimalSize获取mPreviewSize时,您可以交换宽度和高度。 (宽度和高度来自您的视图)。 然后
textureView.setAspectRatio(mPreviewSize.Height, mPreviewSize.Width);
创建预览会话时
texture.SetDefaultBufferSize(mPreviewSize.Width, mPreviewSize.Height);
然后您将预览正常图像
答案 2 :(得分:0)
感谢@Eddy Talvala和@blinker!我带上我的功能可以在任何方向正确更新尺寸:
public void invalidateSize(TextureView textureView) {
Point cameraResolution = getCameraResolution();
int previewRotation = getPreviewRotation();
// Camera and preview on screen in the opposite orientation
if (previewRotation == 90 || previewRotation == 270) {
//noinspection SuspiciousNameCombination
cameraResolution = new Point(cameraResolution.y, cameraResolution.x);
}
float cameraAspectRatio = cameraResolution.x / (float) cameraResolution.y;
float textureAspectRatio = textureView.getWidth() / (float) textureView.getHeight();
float scaleX = cameraAspectRatio / textureAspectRatio;
float scaleY = textureAspectRatio / cameraAspectRatio;
if (scaleX > scaleY) {
scaleY = 1;
} else {
scaleX = 1;
}
Matrix matrix = new Matrix();
matrix.setScale(scaleX, scaleY,
textureView.getWidth() / 2f, textureView.getHeight() / 2f);
textureView.setTransform(matrix);
}