API камеры не работает на KITKAT
У меня действительно странная проблема. Следующий код, который у меня есть, используется для съемки при нажатии кнопки. Он работает правильно на телефонах Jelly Bean, но не на Kitkat:
Основная активность.java :
package com.example.takepic;
import android.app.Activity;
import android.content.pm.PackageManager;
import android.hardware.Camera;
import android.hardware.Camera.CameraInfo;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;
public class MainActivity extends Activity {
private final static String DEBUG_TAG = "MakePhotoActivity";
private Camera camera;
private Button capture = null;
private int cameraId = 0;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
capture = (Button)findViewById(R.id.captureBack);
capture.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
camera.startPreview(); //After this, nothing gets printed, and picture does not get taken
System.out.println("Camera preview has started.");
camera.takePicture(null, null, new PhotoHandler(getApplicationContext()));
}
});
// do we have a camera?
if (!getPackageManager()
.hasSystemFeature(PackageManager.FEATURE_CAMERA)) {
Toast.makeText(this, "No camera on this device", Toast.LENGTH_LONG)
.show();
} else {
cameraId = findBackFacingCamera();
if (cameraId < 0) {
Toast.makeText(this, "No back facing camera found.",
Toast.LENGTH_LONG).show();
} else {
camera = Camera.open(cameraId);
}
}
}
private int findBackFacingCamera() {
int cameraId = -1;
// Search for the front facing camera
int numberOfCameras = Camera.getNumberOfCameras();
for (int i = 0; i < numberOfCameras; i++) {
CameraInfo info = new CameraInfo();
Camera.getCameraInfo(i, info);
if (info.facing == CameraInfo.CAMERA_FACING_BACK) {
Log.d(DEBUG_TAG, "Camera found");
cameraId = i;
break;
}
}
return cameraId;
}
@Override
protected void onPause() {
if (camera != null) {
camera.release();
camera = null;
}
super.onPause();
}
}
Фотохандлер.java :
package com.example.takepic;
import java.io.File;
import java.io.FileOutputStream;
import java.text.SimpleDateFormat;
import java.util.Date;
import android.content.Context;
import android.hardware.Camera;
import android.hardware.Camera.PictureCallback;
import android.os.Environment;
import android.util.Log;
import android.widget.Toast;
public class PhotoHandler implements PictureCallback {
private final Context context;
public PhotoHandler(Context context) {
this.context = context;
}
@Override
public void onPictureTaken(byte[] data, Camera camera) {
File pictureFileDir = getDir();
Toast.makeText(context, "Entered onPictureTaken", Toast.LENGTH_LONG).show();
if (!pictureFileDir.exists() && !pictureFileDir.mkdirs()) {
Log.d("Directory error", "Can't create directory to save image.");
Toast.makeText(context, "Can't create directory to save image.",
Toast.LENGTH_LONG).show();
return;
}
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyymmddhhmmss");
String date = dateFormat.format(new Date());
String photoFile = "Picture_" + date + ".jpg";
String filename = pictureFileDir.getPath() + File.separator + photoFile;
File pictureFile = new File(filename);
try {
FileOutputStream fos = new FileOutputStream(pictureFile);
fos.write(data);
fos.close();
Toast.makeText(context, "New Image saved:" + photoFile,
Toast.LENGTH_LONG).show();
} catch (Exception error) {
Log.d("File saving error", "File" + filename + "not saved: "
+ error.getMessage());
Toast.makeText(context, "Image could not be saved.",
Toast.LENGTH_LONG).show();
}
}
private File getDir() {
// File sdDir = Environment.getExternalStoragePublicDirectory(Environment.DIRECTORY_DOWNLOADS);
File sdDir = new File(Environment.getExternalStorageDirectory().getAbsolutePath());
Toast.makeText(context, ("Path : "+sdDir.getAbsolutePath()), Toast.LENGTH_LONG).show();
return sdDir;
}
}
Файл манифеста:
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
package="com.example.takepic"
android:versionCode="1"
android:versionName="1.0" >
<uses-sdk android:minSdkVersion="15" />
<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
<application
android:allowBackup="true"
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/AppTheme" >
<activity
android:name="com.example.takepic.MainActivity"
android:label="@string/app_name" >
<intent-filter>
<action android:name="android.intent.action.MAIN" />
<category android:name="android.intent.category.LAUNCHER" />
</intent-filter>
</activity>
</application>
Я поместил довольно много тостовых сообщений и операторов печати для отладки. Я не опубликовал logcat здесь, потому что, когда я запускаю это на телефоне KitKat, я получаю ничего на logcat . Никаких исключений или предупреждений.
Когда я запускаю это на jellybean телефоне, он работает правильно, показывая все тосты и печатает и делает снимок.
Когда я запускаю это на Kitkat , я не получаю никаких сообщений отладки после
camera.startPreview();
System.out.println("Camera preview has started.");
Я подозреваю, что проблема связана с API takePicture, но я не могу его отладить.
Пожалуйста, помогите мне решить эту проблему.
EDIT: после дальнейшего анализа я нашел причину для вопрос. ОбъектPhotoHandler вызывается успешно, но методonPictureTaken не вызывается, вероятно, потому, что он не получил информацию о том, что изображение было нажато камерой. Я не знаю почему. Пожалуйста, помогите мне решить эту проблему.
3 ответа:
Я заметил, что вы не назначаете никакого держателя поверхности для камеры. Предоставление поверхности предварительного просмотра для камеры имеет важное значение.
Согласно документам здесь:
Http://developer.android.com/guide/topics/media/camera.html
Следуйте коду, предложенному доком. Фотографирование без предварительного просмотра-это большая проблема безопасности. Парни с андроидами могли бы исправить это в kitkat.
Возможно, вы пропустили эту часть кода при вставке сюда, так что добавлено беспокойство также проверить, что вы выполняете свой код ' камеры.takePicture (null, null, callback) 'внутри метода обратного вызова' onSurfaceCreated ' из SurfaceHolder.
Вы можете получить весь соответствующий код по вышеупомянутой ссылке.
Сборка мусора KitKat работает иначе, чем предыдущие API.
Я предполагаю, что объект
PhotoHandler
, который вы передаете методуtakePicture()
, получает мусор, собранный перед вызовомonPictureTaken
.Попробуйте сделать объект
PhotoHandler
как в переменной экземпляра в вашемMainActivity
.В верхней части класса:
PhotoHandler photoHandler;
Затем в
onCreate()
photoHandler = new PhotoHandler(getApplicationContext());
Затем при вызове
takePicture()
:camera.takePicture(null, null, photoHandler);
Добавьте
uses-feature
в файл манифеста:<uses-feature android:name="android.hardware.camera" /> <uses-feature android:name="android.hardware.camera.autofocus" />
И проверьте эту ссылку. Возможно, это поможет тебе.