#android #camera #sd-card #image-capture
#Android #камера #sd-карта #захват изображения
Вопрос:
эй, ребята, я использую следующий код для доступа к камере из моего приложения. Приложение может получить доступ к камере, я также добавил кнопку, onclicklistener которой добавляет эту строку кода :-
camera.takePicture(mShutterCallback, mPictureCallbackRaw, mPictureCallbackJpeg);
Теперь я не знаю, что происходит, но приложение застревает, я должен принудительно закрыть его, а затем я даже не могу получить доступ к родному приложению камеры.
Я думаю, что он покидает приложение, не отпуская объект камеры.
Кроме того, изображение никогда не сохраняется.
Код :-
SurfaceHolder holder;
SurfaceView surface;
Camera camera;
Boolean isPreviewRunning, fromOnResume;
//Preview mpreview;
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
isPreviewRunning = false;
fromOnResume = false;
// requestWindowFeature(Window.FEATURE_NO_TITLE);
surface = (SurfaceView)findViewById(R.id.surface);
holder = surface.getHolder();
holder.addCallback(this);
holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
Button btn = (Button)findViewById(R.id.click);
btn.setOnClickListener(new OnClickListener(){
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
camera.takePicture(mShutterCallback, mPictureCallbackRaw, mPictureCallbackJpeg);
//onCreate(null);
}
});
}
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width,
int height) {
/* if (isPreviewRunning) {
camera.stopPreview();
} */
Camera.Parameters parameters = camera.getParameters();
List<Size> sizes = parameters.getSupportedPreviewSizes();
Size size = sizes.get(0);
parameters.setPreviewSize(size.width, size.height);
camera.setParameters(parameters);
camera.startPreview();
isPreviewRunning=true;
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
// TODO Auto-generated method stub
camera = Camera.open();
try {
camera.setPreviewDisplay(holder);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
// TODO Auto-generated method stub
camera.stopPreview();
camera.release();
camera=null;
}
Camera.PictureCallback mPictureCallbackRaw = new Camera.PictureCallback() {
public void onPictureTaken(byte[] data, Camera c) {
Log.e(getClass().getSimpleName(), "PICTURE CALLBACK RAW: " data);
camera.startPreview();
}
};
Camera.PictureCallback mPictureCallbackJpeg= new Camera.PictureCallback() {
public void onPictureTaken(byte[] data, Camera c) {
Log.e(getClass().getSimpleName(), "PICTURE CALLBACK JPEG: data.length = " data);
camera.startPreview();
}
};
Camera.ShutterCallback mShutterCallback = new Camera.ShutterCallback() {
public void onShutter() {
Log.e(getClass().getSimpleName(), "SHUTTER CALLBACK");
}
};
/*
* protected void onRestoreInstanceState(Bundle savedInstanceState)
{
super.onRestoreInstanceState(savedInstanceState);
}
protected void onResume()
{
Log.e(getClass().getSimpleName(), "onResume");
camera.open();
fromOnResume=true;
super.onResume();
}
protected void onSaveInstanceState(Bundle outState)
{
super.onSaveInstanceState(outState);
}
@Override
protected void onPause() {
// TODO Auto-generated method stub
camera.release();
super.onPause();
}
protected void onStop()
{
Log.e(getClass().getSimpleName(), "onStop");
super.onStop();
}*/
Ответ №1:
Я знаю, что это не совсем ответ на ваш вопрос, но не проще ли было бы использовать приложение для стандартной камеры? Вы можете получить к нему доступ, используя этот код в своей деятельности:
@Override
public void onCreate(Bundle savedInstanceState)
{
super.onCreate(savedInstanceState);
setContentView(R.layout.capture);
Button capture = (Button) findViewById(R.id.capture_button);
capture.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
// We use the stock camera app to take a photo
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);
intent.putExtra(MediaStore.EXTRA_OUTPUT, getImageUri());
startActivityForResult(intent, TAKE_PHOTO_CODE);
}
});
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == TAKE_PHOTO_CODE amp;amp; resultCode == RESULT_OK) {
Uri imagePath = getImageUri();
doSomething();
}
}
/**
* Get the uri of the captured file
* @return A Uri which path is the path of an image file, stored on the dcim folder
*/
private Uri getImageUri() {
// Store image in dcim
File file = new File(Environment.getExternalStorageDirectory() "/DCIM", CAPTURE_TITLE);
Uri imgUri = Uri.fromFile(file);
return imgUri;
}
Комментарии:
1. Genuin… это решило задачу, большое спасибо … моим следующим шагом будет загрузка изображения на сервер… кроме того, я думаю, что у меня должно быть название изображения в соответствии с датой и временем, разве это не хороший вариант?? потому что в настоящее время все изображения сохраняются только под моим именем 🙂
2. Вам следует проверить этот проект : существует приложение для Android для захвата и загрузки фотографии на удаленный сервер. Код указанного сервера также встроен.
Ответ №2:
Просто совет, используйте EXIF в jpg перед загрузкой на сервер. Я обнаружил, что загрузка изображений утомительно медленная из-за качества современных камер смартфонов. Простым решением является использование программы чтения exif для извлечения миниатюры jpeg, сохранения указанной миниатюры как нового jpeg и ее загрузки. Он идентичен оригинальной фотографии, но намного меньше (менее 100 КБ). Я не уверен, что вам нужно именно такое качество изображения, но если нет, и для загрузки большого количества фотографий используйте метод exif. Я программировал на python sl4a и использовал EXIF.py но я уверен, что в Java есть нечто подобное.