Как захватить изображение и сохранить его на SD-карте

#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 есть нечто подобное.