从相机拍摄照片无需预览

android android-camera android-service

70278 观看

9回复

461 作者的声誉

我正在编写一个Android 1.5应用程序,它在启动后启动。这是一个Service,应该拍照没有预览。此应用程序将记录某些区域的光密度。我能够拍照,但照片是黑色的。

经过长时间的研究,我遇到了一个关于它的bug线程。如果您不生成预览,则图像将为黑色,因为Android相机需要预览才能设置曝光和对焦。我创建了一个SurfaceView和听众,但onSurfaceCreated()事件永远不会被解雇。

我想原因是,表面没有在视觉上创造。我还看到了静态调用相机的一些例子,MediaStore.CAPTURE_OR_SOMETHING它们拍摄照片并用两行代码保存在所需的文件夹中,但它也没有拍照。

我是否需要使用IPC并bindService()调用此功能?或者是否有另一种方法来实现这一目标?

作者: eyurdakul 的来源 发布者: 2010 年 3 月 5 日

回应 (9)


47

1 作者的声誉

决定

Android平台上的摄像头在给出有效的预览表面之前无法流式传输视频真的很奇怪。似乎该平台的架构师根本没有考虑第三方视频流应用程序。即使对于增强现实案例,图片也可以呈现为某种视觉替代,而不是实时相机流。

无论如何,您只需将预览曲面的大小调整为1x1像素,并将其放在窗口小部件的角落(视觉元素)。请注意 - 调整预览面大小,而不是相机框架尺寸。

当然,这种技巧并不能消除消耗一些系统资源和电池的不需要的数据流(用于预览)。

作者: alexander.o.krutyakov 发布者: 07.10.2010 11:10

36

1051 作者的声誉

实际上它是可能的,但您必须使用虚拟SurfaceView伪造预览

SurfaceView view = new SurfaceView(this);
c.setPreviewDisplay(view.getHolder());
c.startPreview();
c.takePicture(shutterCallback, rawPictureCallback, jpegPictureCallback);

更新9/21/11:显然这对每个Android设备都不起作用。

作者: Frank 发布者: 21.03.2011 11:23

13

196 作者的声誉

我们通过在3.0以下版本中使用虚拟SurfaceView(未添加到实际GUI)解决了这个问题(或者说4.0作为平板电脑上的摄像头服务并不真正有意义)。在版本> = 4.0中,这只在模拟器中工作;(使用SurfaceTexture(和setSurfaceTexture())而不是SurfaceView(和setSurfaceView())在这里工作。至少这适用于Nexus S.

我认为这确实是Android框架的一个缺点。

作者: mnl 发布者: 20.04.2012 03:05

20

483 作者的声誉

有一种方法可以做到这一点,但它有点棘手。应该做什么,是从服务附加一个Surfaceholder到窗口管理器

WindowManager wm = (WindowManager) mCtx.getSystemService(Context.WINDOW_SERVICE);
params = new WindowManager.LayoutParams(WindowManager.LayoutParams.WRAP_CONTENT,
            WindowManager.LayoutParams.WRAP_CONTENT,
            WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
            WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
            PixelFormat.TRANSLUCENT);        
wm.addView(surfaceview, params);

然后设置

surfaceview.setZOrderOnTop(true);
mHolder.setFormat(PixelFormat.TRANSPARENT);

其中mHolder是您从表面视图获得的持有者。

这样,您可以使用surfaceview的alpha,使其完全透明,但相机仍然会获得帧。

这就是我的表现。希望能帮助到你 :)

作者: Vlad 发布者: 22.04.2012 02:12

38

2828 作者的声誉

我在Android Camera Docs中找到了答案。

注意:可以先使用MediaRecorder而不先创建相机预览,然后跳过此过程的前几个步骤。但是,由于用户通常更喜欢在开始录制之前看到预览,因此这里不讨论该过程。

您可以在上面的链接中找到分步说明。在说明之后,它将说明我在上面提供的报价。

作者: Phillip Scott Givens 发布者: 06.06.2012 06:57

20

467 作者的声誉

在Android 4.0及更高版本(API级别> = 14),您可以使用TextureView预览相机流并使其不可见,以便不向用户显示。这是如何做:

首先创建一个类来实现SurfaceTextureListener,它将获得预览表面的创建/更新回调。该类还将摄像机对象作为输入,以便在创建曲面后立即调用摄像机的startPreview函数:

public class CamPreview extends TextureView implements SurfaceTextureListener {

  private Camera mCamera;

  public CamPreview(Context context, Camera camera) {
    super(context);
    mCamera = camera;
   }

  @Override
  public void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) {
    Camera.Size previewSize = mCamera.getParameters().getPreviewSize();
    setLayoutParams(new FrameLayout.LayoutParams(
        previewSize.width, previewSize.height, Gravity.CENTER));

    try{
      mCamera.setPreviewTexture(surface);
     } catch (IOException t) {}

    mCamera.startPreview();
    this.setVisibility(INVISIBLE); // Make the surface invisible as soon as it is created
  }

  @Override
  public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) {
      // Put code here to handle texture size change if you want to
  }

  @Override
  public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
    return true;
  }

  @Override
  public void onSurfaceTextureUpdated(SurfaceTexture surface) {
      // Update your view here!
  }
}

您还需要实现一个回调类来处理预览数据:

public class CamCallback implements Camera.PreviewCallback{
  public void onPreviewFrame(byte[] data, Camera camera){
     // Process the camera data here
  }
}

使用上面的CamPreview和CamCallback类在活动的onCreate()或类似的启动函数中设置相机:

// Setup the camera and the preview object
Camera mCamera = Camera.open(0);
CamPreview camPreview = new CamPreview(Context,mCamera);
camPreview.setSurfaceTextureListener(camPreview);

// Connect the preview object to a FrameLayout in your UI
// You'll have to create a FrameLayout object in your UI to place this preview in
FrameLayout preview = (FrameLayout) findViewById(R.id.cameraView); 
preview.addView(camPreview);

// Attach a callback for preview
CamCallback camCallback = new CamCallback();
mCamera.setPreviewCallback(camCallback);
作者: Varun Gulshan 发布者: 09.01.2013 02:43

34

22361 作者的声誉

拍照

在尝试隐藏预览之前先使其工作。

  • 正确设置预览
    • 使用SurfaceView(Android-4.0之前的兼容性)或SurfaceTexture(Android 4+,可以透明)
    • 在拍照之前设置并初始化它
    • 等待SurfaceViewSurfaceHolder(通过getHolder())报告surfaceCreated()TextureView报告onSurfaceTextureAvailableSurfaceTextureListener 设置和初始化前预览。
  • 确保预览可见:
    • 把它添加到 WindowManager
    • 确保其布局大小至少为1x1像素(您可能希望首先将其设为MATCH_PARENTx MATCH_PARENT进行测试)
    • 确保其可见性View.VISIBLE(如果您未指定它,这似乎是默认值)
    • 确保您使用FLAG_HARDWARE_ACCELERATEDLayoutParams,如果这是一个TextureView
  • 使用takePicture的是JPEG回调,因为文档说所有设备都不支持其他回调

故障排除

  • 如果surfaceCreated/ onSurfaceTextureAvailable没有被调用时,SurfaceView/ TextureView可能不被显示。
  • 如果takePicture失败,请首先确保预览正常。您可以移除takePicture呼叫并让预览运行以查看它是否显示在屏幕上。
  • 如果图像比应该更暗,则可能需要在呼叫前延迟约一秒钟,takePicture以便在预览开始后相机有时间调整其曝光。

隐藏预览

  • 预览View1x1尺寸以最小化其可见性(或尝试8x16以获得更高的可靠性

    new WindowManager.LayoutParams(1, 1, /*...*/)
    
  • 将预览移出中心以降低其醒目性:

    new WindowManager.LayoutParams(width, height,
        Integer.MIN_VALUE, Integer.MIN_VALUE, /*...*/)
    
  • 使预览透明(仅适用于TextureView

    WindowManager.LayoutParams params = new WindowManager.LayoutParams(
        width, height, /*...*/
        PixelFormat.TRANSPARENT);
    params.alpha = 0;
    

工作示例(在Sony Xperia M,Android 4.3上测试)

/** Takes a single photo on service start. */
public class PhotoTakingService extends Service {

    @Override
    public void onCreate() {
        super.onCreate();
        takePhoto(this);
    }

    @SuppressWarnings("deprecation")
    private static void takePhoto(final Context context) {
        final SurfaceView preview = new SurfaceView(context);
        SurfaceHolder holder = preview.getHolder();
        // deprecated setting, but required on Android versions prior to 3.0
        holder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);

        holder.addCallback(new Callback() {
            @Override
            //The preview must happen at or after this point or takePicture fails
            public void surfaceCreated(SurfaceHolder holder) {
                showMessage("Surface created");

                Camera camera = null;

                try {
                    camera = Camera.open();
                    showMessage("Opened camera");

                    try {
                        camera.setPreviewDisplay(holder);
                    } catch (IOException e) {
                        throw new RuntimeException(e);
                    }

                    camera.startPreview();
                    showMessage("Started preview");

                    camera.takePicture(null, null, new PictureCallback() {

                        @Override
                        public void onPictureTaken(byte[] data, Camera camera) {
                            showMessage("Took picture");
                            camera.release();
                        }
                    });
                } catch (Exception e) {
                    if (camera != null)
                        camera.release();
                    throw new RuntimeException(e);
                }
            }

            @Override public void surfaceDestroyed(SurfaceHolder holder) {}
            @Override public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {}
        });

        WindowManager wm = (WindowManager)context
            .getSystemService(Context.WINDOW_SERVICE);
        WindowManager.LayoutParams params = new WindowManager.LayoutParams(
                1, 1, //Must be at least 1x1
                WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
                0,
                //Don't know if this is a safe default
                PixelFormat.UNKNOWN);

        //Don't set the preview visibility to GONE or INVISIBLE
        wm.addView(preview, params);
    }

    private static void showMessage(String message) {
        Log.i("Camera", message);
    }

    @Override public IBinder onBind(Intent intent) { return null; }
}
作者: Sam 发布者: 22.11.2014 11:43

3

31 作者的声誉

在“Sam的工作示例”中(谢谢Sam ......)

如果在istruction“wm.addView(预览,params);”

获取异常“无法添加窗口android.view.ViewRoot - 此窗口类型的权限被拒绝”

在AndroidManifest中使用此权限解决:

<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
作者: Giulio 发布者: 08.01.2016 01:28

1

1207 作者的声誉

你可以尝试这个工作代码,这个服务点击前面的图片,如果你想捕获相机图片然后在代码和注释frontCamera取消注释backCamera。

注意: - 允许来自App的Camera和Storage权限以及来自Activity或任何地方的startService。

public class MyService extends Service {

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public void onCreate() {
        super.onCreate();
        CapturePhoto();
    }

    private void CapturePhoto() {

        Log.d("kkkk","Preparing to take photo");
        Camera camera = null;

        Camera.CameraInfo cameraInfo = new Camera.CameraInfo();

            int frontCamera = 1;
            //int backCamera=0;

            Camera.getCameraInfo(frontCamera, cameraInfo);

            try {
                camera = Camera.open(frontCamera);
            } catch (RuntimeException e) {
                Log.d("kkkk","Camera not available: " + 1);
                camera = null;
                //e.printStackTrace();
            }
            try {
                if (null == camera) {
                    Log.d("kkkk","Could not get camera instance");
                } else {
                    Log.d("kkkk","Got the camera, creating the dummy surface texture");
                     try {
                         camera.setPreviewTexture(new SurfaceTexture(0));
                        camera.startPreview();
                    } catch (Exception e) {
                        Log.d("kkkk","Could not set the surface preview texture");
                        e.printStackTrace();
                    }
                    camera.takePicture(null, null, new Camera.PictureCallback() {

                        @Override
                        public void onPictureTaken(byte[] data, Camera camera) {
                            File pictureFileDir=new File("/sdcard/CaptureByService");

                            if (!pictureFileDir.exists() && !pictureFileDir.mkdirs()) {
                                pictureFileDir.mkdirs();
                            }
                            SimpleDateFormat dateFormat = new SimpleDateFormat("yyyymmddhhmmss");
                            String date = dateFormat.format(new Date());
                            String photoFile = "ServiceClickedPic_" + "_" + date + ".jpg";
                            String filename = pictureFileDir.getPath() + File.separator + photoFile;
                            File mainPicture = new File(filename);

                            try {
                                FileOutputStream fos = new FileOutputStream(mainPicture);
                                fos.write(data);
                                fos.close();
                                Log.d("kkkk","image saved");
                            } catch (Exception error) {
                                Log.d("kkkk","Image could not be saved");
                            }
                            camera.release();
                        }
                    });
                }
            } catch (Exception e) {
                camera.release();
            }
    }
}
作者: kdblue 发布者: 03.01.2018 09:31
32x32