LoginSignup
19
19

More than 5 years have passed since last update.

今更カメラ制御の基礎(STEP3:撮影?して保存)

Last updated at Posted at 2014-10-18

はじめに

このサンプルを実行すると、Preview画面が90度ずれたりします。が、それらへの対応処理を入れると可読性が落ちるのでとりあえず無視。
STEP4で直す予定(取り急ぎ、STEP3.5を下の方で紹介)。

やりたいこと

とりあえず画像を保存する。ただ、今回はtakePicture()は使わない。

STEP2からの追加

・基本的にsetOneShotPreviewCallback内の実装が異なるだけ。

他のステップはこちら

STEP1:基礎
STEP2:オートフォーカスとか
STEP3:保存
STEP4:アスペクト比調整

ハマりどころ

・onPreviewFrameで受け取ったbyte[]はYUV形式なので、直接保存できない。
・YUV⇒JPG⇒BMPと変換してから保存する必要があり、ややこしい(別のいい方法が無いのか?)

参考サイト

YUV−BMP変換

ソース

MyActivity.java
public class MyActivity extends Activity {

    private SurfaceView mySurfaceView;
    private Camera myCamera; //hardware


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_my);

        //SurfaceView
        mySurfaceView = (SurfaceView)findViewById(R.id.mySurfaceVIew);
        //リスナ追加
        mySurfaceView.setOnClickListener(onSurfaceClickListener);

        //SurfaceHolder(SVの制御に使うInterface)
        SurfaceHolder holder = mySurfaceView.getHolder();
        //コールバックを設定
        holder.addCallback(callback);

    }

    //コールバック
    private SurfaceHolder.Callback callback = new SurfaceHolder.Callback() {
        @Override
        public void surfaceCreated(SurfaceHolder surfaceHolder) {

            //CameraOpen
            myCamera = Camera.open();

            //出力をSurfaceViewに設定
            try{
                myCamera.setPreviewDisplay(surfaceHolder);
            }catch(Exception e){
                e.printStackTrace();
            }

        }

        @Override
        public void surfaceChanged(SurfaceHolder surfaceHolder, int i, int i2, int i3) {

            //プレビュースタート(Changedは最初にも1度は呼ばれる)
            myCamera.startPreview();

        }

        @Override
        public void surfaceDestroyed(SurfaceHolder surfaceHolder) {

            //片付け
            myCamera.release();
            myCamera = null;
        }
    };

    //Surfaceをクリックした時
    private View.OnClickListener onSurfaceClickListener = new View.OnClickListener(){
        @Override
        public void onClick(View view){
            if(myCamera != null){
                //AutoFocusを実行
                myCamera.autoFocus(autoFocusCallback);
            }
        }

    };

    //AutoFocusした時
    private Camera.AutoFocusCallback autoFocusCallback = new Camera.AutoFocusCallback(){
        @Override
        public void onAutoFocus(boolean b, Camera camera) {

            //ただ写真を撮るだけならここにtakePicture()を入れても良い
            //しかし、Pixelを弄りたいのでもうワンクッション(画像処理やバーコード処理をしたいので)
            //Previewを1枚切り取る(そしてイベント発生)
            camera.setOneShotPreviewCallback(previewCallback);
        }
    };

    //切り取った時(ここで撮影、各種画像処理を行う)
    private Camera.PreviewCallback previewCallback = new Camera.PreviewCallback(){

        //OnShotPreview時のbyte[]が渡ってくる
        @Override
        public void onPreviewFrame(byte[] bytes, Camera camera) {

            //STEP3

            //プレビューのフォーマットはYUVなので、YUVをBmpに変換する必要がある(ややこしい)
            int w = camera.getParameters().getPreviewSize().width;
            int h = camera.getParameters().getPreviewSize().height;

            //切り取った画像を保存
            //Bitmap bmp = BitmapFactory.decodeByteArray(bytes,0,bytes.length,null);
            Bitmap bmp = getBitmapImageFromYUV(bytes,w,h);
             //Uriが戻る
            String uri = MediaStore.Images.Media.insertImage(getContentResolver(),bmp,"",null);

        }
    };

    //YUVをBitmapに変換する関数(参照:http://tech.thecoolblogs.com/2013/02/get-bitmap-image-from-yuv-in-android.html)
    public static Bitmap getBitmapImageFromYUV(byte[] data, int width, int height) {
        YuvImage yuvimage = new YuvImage(data, ImageFormat.NV21, width, height, null);
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        yuvimage.compressToJpeg(new Rect(0, 0, width, height), 80, baos);
        byte[] jdata = baos.toByteArray();
        BitmapFactory.Options bitmapFatoryOptions = new BitmapFactory.Options();
        bitmapFatoryOptions.inPreferredConfig = Bitmap.Config.RGB_565;
        Bitmap bmp = BitmapFactory.decodeByteArray(jdata, 0, jdata.length, bitmapFatoryOptions);
        return bmp;
    }

}

STEP3.5 取り急ぎ、Portrait対応

とりあえず縦対応。フロントカメラじゃないと行けないと思う。Nexus7で確認。ただ、アスペクト比はおかしい。

上のコードの違い

ディスプレイを90度回転

myCamera.setDisplayOrientation(90);

保存する画像を回転。

@Override
        public void onPreviewFrame(byte[] bytes, Camera camera) {

            //STEP3

            //プレビューのフォーマットはYUVなので、YUVをBmpに変換する必要がある(ややこしい)
            int w = camera.getParameters().getPreviewSize().width;
            int h = camera.getParameters().getPreviewSize().height;

            //切り取った画像を保存
            //Bitmap bmp = BitmapFactory.decodeByteArray(bytes,0,bytes.length,null);
            Bitmap bmp = getBitmapImageFromYUV(bytes, w, h);

            //回転
            Matrix m = new Matrix();
            m.setRotate(90);

            //保存用 bitmap生成
            Bitmap rotated_bmp = Bitmap.createBitmap(bmp,0,0,bmp.getWidth(),bmp.getHeight(),m,true);

            //保存
            MediaStore.Images.Media.insertImage(getContentResolver(),rotated_bmp,"",null);

        }

ソース

MyActivity.java
public class MyActivity extends Activity {

    private SurfaceView mySurfaceView;
    private Camera myCamera; //hardware


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_my);

        //SurfaceView
        mySurfaceView = (SurfaceView)findViewById(R.id.mySurfaceVIew);
        //リスナ追加
        mySurfaceView.setOnClickListener(onSurfaceClickListener);

        //SurfaceHolder(SVの制御に使うInterface)
        SurfaceHolder holder = mySurfaceView.getHolder();
        //コールバックを設定
        holder.addCallback(callback);

    }

    //コールバック
    private SurfaceHolder.Callback callback = new SurfaceHolder.Callback() {
        @Override
        public void surfaceCreated(SurfaceHolder surfaceHolder) {

            //CameraOpen
            myCamera = Camera.open();

            //Portrait対応
            myCamera.setDisplayOrientation(90);

            //出力をSurfaceViewに設定
            try{
                myCamera.setPreviewDisplay(surfaceHolder);
            }catch(Exception e){
                e.printStackTrace();
            }

        }

        @Override
        public void surfaceChanged(SurfaceHolder surfaceHolder, int i, int i2, int i3) {

            //プレビュースタート(Changedは最初にも1度は呼ばれる)
            myCamera.startPreview();

        }

        @Override
        public void surfaceDestroyed(SurfaceHolder surfaceHolder) {

            //片付け
            myCamera.release();
            myCamera = null;
        }
    };

    //Surfaceをクリックした時
    private View.OnClickListener onSurfaceClickListener = new View.OnClickListener(){
        @Override
        public void onClick(View view){
            if(myCamera != null){
                //AutoFocusを実行
                myCamera.autoFocus(autoFocusCallback);
            }
        }

    };

    //AutoFocusした時
    private Camera.AutoFocusCallback autoFocusCallback = new Camera.AutoFocusCallback(){
        @Override
        public void onAutoFocus(boolean b, Camera camera) {

            //ただ写真を撮るだけならここにtakePicture()を入れても良い
            //しかし、Pixelを弄りたいのでもうワンクッション(画像処理やバーコード処理をしたいので)
            //Previewを1枚切り取る(そしてイベント発生)
            camera.setOneShotPreviewCallback(previewCallback);
        }
    };

    //切り取った時(ここで撮影、各種画像処理を行う)
    private Camera.PreviewCallback previewCallback = new Camera.PreviewCallback(){

        //OnShotPreview時のbyte[]が渡ってくる
        @Override
        public void onPreviewFrame(byte[] bytes, Camera camera) {

            //STEP3

            //プレビューのフォーマットはYUVなので、YUVをBmpに変換する必要がある(ややこしい)
            int w = camera.getParameters().getPreviewSize().width;
            int h = camera.getParameters().getPreviewSize().height;

            //切り取った画像を保存
            //Bitmap bmp = BitmapFactory.decodeByteArray(bytes,0,bytes.length,null);
            Bitmap bmp = getBitmapImageFromYUV(bytes, w, h);

            //回転
            Matrix m = new Matrix();
            m.setRotate(90);

            //保存用 bitmap生成
            Bitmap rotated_bmp = Bitmap.createBitmap(bmp,0,0,bmp.getWidth(),bmp.getHeight(),m,true);

            //保存
            MediaStore.Images.Media.insertImage(getContentResolver(),rotated_bmp,"",null);

        }
    };

    //YUVをBitmapに変換する関数(参照:http://tech.thecoolblogs.com/2013/02/get-bitmap-image-from-yuv-in-android.html)
    public static Bitmap getBitmapImageFromYUV(byte[] data, int width, int height) {
        YuvImage yuvimage = new YuvImage(data, ImageFormat.NV21, width, height, null);
        ByteArrayOutputStream baos = new ByteArrayOutputStream();
        yuvimage.compressToJpeg(new Rect(0, 0, width, height), 80, baos);
        byte[] jdata = baos.toByteArray();
        BitmapFactory.Options bitmapFatoryOptions = new BitmapFactory.Options();
        bitmapFatoryOptions.inPreferredConfig = Bitmap.Config.RGB_565;
        Bitmap bmp = BitmapFactory.decodeByteArray(jdata, 0, jdata.length, bitmapFatoryOptions);
        return bmp;
    }

}
19
19
1

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
19
19