Android 自定义Camera(一)

作者: jinguangyue | 来源:发表于2016-04-15 11:40 被阅读8211次

    公司一直在做一款好玩的应用名叫Funny, 是一个以萌和可爱为主的图片社区,P图工具(欢迎大家下载玩耍哈),刚开始调用的相机是调用系统的相机, 这肯定不能满足一款应用的需求,So我开始了从零的自定义相机,发现github csdn等等上面关于自定义相机的资料不是很多,当然也有比较好的,这里我也是从开始能预览到后面的一些细节处理踩了一些坑,我会一步一步的分享粗来下面就开始从头开始做个自己的相机吧。

    获取相机和释放相机

    首先相机实例只能有一个,拍照之后一定要释放,那么它也有了自己的生命周期, 首先获取相机,这里传入一个ID,这个id可以设前置摄像头和后置摄像头

    /**
         * 获取Camera实例
         * @return
         */
        private Camera getCamera(int id){
            Camera camera = null;
            try{
                camera = Camera.open(id);
            }catch (Exception e){
    
            }
            return camera;
        }
    
    /**
         * 释放相机资源
         */
        private void releaseCamera(){
            if(mCamera != null){
                mCamera.setPreviewCallback(null);
                mCamera.stopPreview();
                mCamera.release();
                mCamera = null;
            }
        }
    
    /**
         * 预览相机
         */
        private void startPreview(Camera camera, SurfaceHolder holder){
            try {
            //这里要设置相机的一些参数,下面会详细说下
                setupCamera(camera);
                camera.setPreviewDisplay(holder);
                //亲测的一个方法 基本覆盖所有手机 将预览矫正
                CameraUtil.getInstance().setCameraDisplayOrientation(this, mCameraId, camera);
    //            camera.setDisplayOrientation(90);
                camera.startPreview();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    

    使用之后释放相机资源,这里这样写是有原因的,一步都不能少,我有时候会忘记写mCamera.setPreviewCallback(null); 会报Method called after release 错误, 因为相机要实时的预览,那普通的View就不能满足绘制的要求了,这里要用的一个双缓冲机制的SurfaceView,下面我们要做的就是将Camera和SurfaceView绑定起来, 那就需要另一个类了SurfaceHolder, **这里要特别提的是CameraUtil.getInstance().setCameraDisplayOrientation(this, mCameraId, camera); **后面我会把源码地址写上, 由于安卓系统默认预览都是横着的,基本操作的是camera.setDisplayOrientation(90);直接旋转90度矫正, 但是本人在开发的时候就是遇到一些手机上预览反过来了, 最后用这个方法解决了, 它的基本原理就是有的手机系统底层对预览进行了矫正,有的没有,那通过cameraInfo可以判断到。SO这个完美的解决了我的问题, 希望也能帮到你.

    设置SurfaceHolder

    surfaceView = (SurfaceView) findViewById(R.id.surfaceView);
    mHolder = surfaceView.getHolder();
    mHolder.addCallback(this);
    

    这里SurfaceHolder要添加一个回调方法,看注释

    @Override
    public void surfaceCreated(SurfaceHolder holder) {
     //在surface创建的时候开启相机预览
     startPreview(mCamera, holder);
    }
    
    @Override
    public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
     //在相机改变的时候调用此方法, 此时应该先停止预览, 然后重新启动
     mCamera.stopPreview();
     startPreview(mCamera, holder);
    }
    
    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {
     //在destroy的时候释放相机资源
     releaseCamera();
    }
    

    这里要详细说下setCamera方法,不加这个方法当然也可以进行正常的预览, 但是这个方法要设置相机的预览尺寸,返回图片尺寸,特别要注意的是,预览尺寸返回图片尺寸和surfaceView的尺寸比例必须是一样的,要不预览或者拍照返回的图片会变形或者程序崩溃, 下面看代码:

    /**
     * 设置
     */
    private void setupCamera(Camera camera) {
     Camera.Parameters parameters = camera.getParameters();
    
     List < String > focusModes = parameters.getSupportedFocusModes();
     if (focusModes.contains(Camera.Parameters.FOCUS_MODE_AUTO)) {
      // Autofocus mode is supported 自动对焦
      parameters.setFocusMode(Camera.Parameters.FOCUS_MODE_AUTO);
     }
    
     //这里第三个参数为最小尺寸 getPropPreviewSize方法会对从最小尺寸开始升序排列 取出所有支持尺寸的最小尺寸
     Camera.Size previewSize = CameraUtil.getInstance().getPropPreviewSize(parameters.getSupportedPreviewSizes(), 1000);
     parameters.setPreviewSize(previewSize.width, previewSize.height);
    
     Camera.Size pictrueSize = CameraUtil.getInstance().getPropPictureSize(parameters.getSupportedPictureSizes(), 1000);
     parameters.setPictureSize(pictrueSize.width, pictrueSize.height);
    
     camera.setParameters(parameters);
    
     Log.d("previewSize.width===", previewSize.width + "");
     Log.d("previewSize.height===", previewSize.height + "");
    
     /**
      * 设置surfaceView的尺寸 因为camera默认是横屏,所以取得支持尺寸也都是横屏的尺寸
      * 我们在startPreview方法里面把它矫正了过来,但是这里我们设置设置surfaceView的尺寸的时候要注意 previewSize.height<previewSize.width
      * previewSize.width才是surfaceView的高度
      * 一般相机都是屏幕的宽度 这里设置为屏幕宽度 高度自适应 你也可以设置自己想要的大小
      */
     FrameLayout.LayoutParams params = new FrameLayout.LayoutParams(screenWidth, screenWidth * previewSize.width / previewSize.height);
     //这里当然可以设置拍照位置 比如居中 我这里就置顶了
     //params.gravity = Gravity.CENTER;
     surfaceView.setLayoutParams(params);
    }
    

    这里还有个地方要特别注意,就是你设置预览尺寸和返回图片尺寸必须要是当前手机支持的尺寸,这里通过getPropPictureSize和getPropPreviewSize获取所有手机支持的尺寸,默认的相机预览都是横屏的, 这里设置一个最小的预览宽度1000, 只要大于1000的支持尺寸都可以。 这里还有最后一步也是非常的重要, 就是Camera的生命周期一定要和当前Activity绑定起来, 所以我们这样做:

    @Override
    protected void onResume() {
     super.onResume();
     if (mCamera == null) {
      mCamera = getCamera(mCameraId);
      if (mHolder != null) {
       startPreview(mCamera, mHolder);
      }
     }
    }
    
    @Override
    protected void onPause() {
     super.onPause();
     releaseCamera();
    }
    

    这里在Activity要展示的时候启动相机预览,在Activity onPause的时候释放相机资源就可以了。那现在可以预览了也要可以拍照哇,接下来来个拍照方法:

    private void captrue() {
     //参数说明 
     mCamera.takePicture(null, null, new Camera.PictureCallback() {
      @Override
      public void onPictureTaken(byte[] data, Camera camera) {
       //将data 转换为位图 或者你也可以直接保存为文件使用 FileOutputStream
       //这里我相信大部分都有其他用处把 比如加个水印 后续再讲解
       Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
       CameraUtil.getInstance().setTakePicktrueOrientation(mCameraId, bitmap);
       //这里打印宽高 就能看到 CameraUtil.getInstance().getPropPictureSize(parameters.getSupportedPictureSizes(), 200);
       // 这设置的最小宽度影响返回图片的大小 所以这里一般这是1000左右把我觉得
       Log.d("bitmapWidth==", bitmap.getWidth() + "");
       Log.d("bitmapHeight==", bitmap.getHeight() + "");
      }
     });
    }
    

    这里有个地方要说一下,上面我们对预览进行了矫正, 那生成的图片我们也要矫正回来啊, 执行CameraUtil.getInstance().setTakePicktrueOrientation(mCameraId, bitmap); 当然也是通过android.hardware.Camera.CameraInfo来操作的。

    我觉得一篇文章不宜太长,太长看着头疼哈哈,那到这里一个简单的demo差不多就可以了,其实要注意的地方也挺多的,我都是一个坑一个坑踩过去的,后面我会将切换前后置摄像头,闪光灯模式,延迟拍摄,正方形取景框,加水印效果等等吧陆续的写出来.

    下面是源码地址
    https://github.com/jinguangyue/CustomCamera
    Star一下并不亏.

    相关文章

      网友评论

      • 33af5afb733e:调个照相机 画面真是花呀
      • 求知131525:大牛我想问下你是在哪里设置的横竖屏转向时图片不变了,能给下qq不,我qq是2464908827,希望博主加下,在线等
      • zziazm:你好,我用Camera.open(0)返回值是null可能是什么情况导致的?
        ckwcc:可能是没有动态申请拍照权限吧,老铁
      • 男小北:如果要手动对焦 该怎么操作呢
      • Shawn_GBWang:您好,我试了一下您的这个dmeo 不能对焦,请问什么问题呢?
        jinguangyue:@Shawn_GBWang 对准一个地方不变 会自动对焦
      • 徐凤:你好,你这个我直接将CameraActivity在配置文件里面设置为横屏,预览试图就变形,应该怎么解决呀
        jinguangyue:@徐凤 目前我这个不支持横屏, 你想横屏的话主要研究setupcamera方法, 里面获得相机预览尺寸和支持尺寸
      • 张贤同学:这博客里贴的代码也是醉了,可以贴全一点吗?照着敲一堆错
        jinguangyue:@来删我 https://github.com/jinguangyue/CustomCamera 有源码地址
      • 5c86e01fae2c:学习一下
      • 尹star: :clap:
        jinguangyue: @尹star 😀😀😀

      本文标题:Android 自定义Camera(一)

      本文链接:https://www.haomeiwen.com/subject/hpcnlttx.html