工模-Camera测试

本文介绍了一个基于Android的应用程序,该程序能够自动循环开启摄像头并拍照。应用通过检查摄像头硬件、选择合适的摄像头,并利用SurfaceView预览,实现了定时自动对焦及拍照的功能。

该模块的主要功能要求是实现自动循环开启摄像头并拍照。
主要实现代码如下:

package com.dw.cameratest2;
import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.Timer;
import java.util.TimerTask;
import android.annotation.SuppressLint;
import android.app.Activity;
import android.content.Context;
import android.content.pm.PackageManager;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Matrix;
import android.hardware.Camera;
import android.hardware.Camera.AutoFocusCallback;
import android.hardware.Camera.PictureCallback;
import android.os.Bundle;
import android.os.Environment;
import android.util.Log;
import android.view.SurfaceHolder;
import android.view.SurfaceView;
import android.view.Window;
import android.view.WindowManager;
import android.widget.Toast;

/**
 * 
 * @author  dw
 * 
 */
public class CameraTestActivity extends Activity {
    private static final String TAG="com.dw.cameratest2.CameraTestActivity";
    private SurfaceView mySurfaceView;
    private SurfaceHolder myHolder;
    private Camera myCamera;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        // 无title
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        // 全屏
        getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
                WindowManager.LayoutParams.FLAG_FULLSCREEN);
        // 设置布局
        setContentView(R.layout.activity_camera_test);
        // 初始化surface
        initSurface();
        // 这里得开线程进行拍照,因为Activity还未完全显示的时候,是无法进行拍照的,SurfaceView必须先显示
        new Thread(new Runnable() {
            @Override
            public void run() {
                // 初始化camera并对焦拍照
                initCamera();
            }
        }).start();

    }

    // 初始化surface
    @SuppressWarnings("deprecation")
    private void initSurface() {
        // 初始化surfaceview
        mySurfaceView = (SurfaceView) findViewById(R.id.camera_surfaceview);

        // 初始化surfaceholder
        myHolder = mySurfaceView.getHolder();
        myHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);

    }
    // 初始化摄像头
    private void initCamera() {
        // 如果存在摄像头
        if (checkCameraHardware(getApplicationContext())) {
            // 获取摄像头(首选前置,无前置选后置)
            if (openFacingFrontCamera()) {
                // 进行对焦
                autoFocus();
            } else {
                Log.d("Demo", "openCameraFailed");
            }

        }
    }

    // 对焦并拍照
    private void autoFocus() {
        try {
            // 因为开启摄像头需要时间,这里让线程延迟2s
            Thread.sleep(2000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        Timer timer = new Timer();
        TimerTask task = new TimerTask() {
            @Override
            public void run() {
                // TODO Auto-generated method stub
                // 自动对焦
                myCamera.autoFocus(myAutoFocus);
                // 对焦后拍照
                myCamera.takePicture(null, null, myPicCallback);
            }
        };
        timer.schedule(task, 1000, 10000);


    }

    // 判断是否存在摄像头
    private boolean checkCameraHardware(Context context) {

        if (context.getPackageManager().hasSystemFeature(
                PackageManager.FEATURE_CAMERA)) {
            // 设备存在摄像头
            return true;
        } else {
            // 设备不存在摄像头
            Log.e(TAG, "未检测到摄像头");
            return false;
        }
    }

    // 获取摄像头
    @SuppressLint("NewApi")
    private boolean openFacingFrontCamera() {

        // 尝试开启后置摄像头
        Camera.CameraInfo cameraInfo = new Camera.CameraInfo();
        for (int camIdx = 0, cameraCount = Camera.getNumberOfCameras(); camIdx < cameraCount; camIdx++) {
            Camera.getCameraInfo(camIdx, cameraInfo);
            if (cameraInfo.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
                try {
                    Log.d("Demo", "tryToOpenCamera");
                    myCamera = Camera.open(camIdx);
                } catch (RuntimeException e) {
                    e.printStackTrace();
                    return false;
                }
            }
        }
        // 如果开启后置失败(或无后置)则开启前置
        if (myCamera == null) {
            for (int camIdx = 0, cameraCount = Camera.getNumberOfCameras(); camIdx < cameraCount; camIdx++) {
                Camera.getCameraInfo(camIdx, cameraInfo);
                if (cameraInfo.facing == Camera.CameraInfo.CAMERA_FACING_BACK) {
                    try {
                        myCamera = Camera.open(camIdx);
                    } catch (RuntimeException e) {
                        return false;
                    }
                }
            }
        }
        try {
            // 这里的myCamera为已经初始化的Camera对象
            myCamera.setPreviewDisplay(myHolder);
        } catch (IOException e) {
            e.printStackTrace();
            myCamera.stopPreview();
            myCamera.release();
            myCamera = null;
        }
        myCamera.startPreview();
        return true;
    }
    // 自动对焦回调函数(空实现)
    private AutoFocusCallback myAutoFocus = new AutoFocusCallback() {
        @Override
        public void onAutoFocus(boolean success, Camera camera) {
        }
    };
    // 拍照成功回调函数
    private PictureCallback myPicCallback = new PictureCallback() {
        @Override
        public void onPictureTaken(byte[] data, Camera camera) {
            // 完成拍照后关闭Activity
            //CameraTestActivity.this.finish();
            // 将得到的照片进行270°旋转,使其竖直
            Bitmap bitmap = BitmapFactory.decodeByteArray(data, 0, data.length);
            Matrix matrix = new Matrix();
            matrix.preRotate(270);
            bitmap = Bitmap.createBitmap(bitmap, 0, 0, bitmap.getWidth(),
                    bitmap.getHeight(), matrix, true);
            // 创建并保存图片文件,每拍一次会覆盖上一次拍的照片以节省空间
            File pictureFile = new File(getDir(), "camera_test.jpeg");
            try {
                FileOutputStream fos = new FileOutputStream(pictureFile);
                bitmap.compress(Bitmap.CompressFormat.PNG, 100, fos);
                fos.close();
            } catch (Exception error) {
                Toast.makeText(CameraTestActivity.this, "拍照失败",
                        Toast.LENGTH_SHORT).show();
                ;
                Log.d("Demo", "保存照片失败" + error.toString());
                error.printStackTrace();
                myCamera.stopPreview();
                myCamera.release();
                myCamera = null;
            }
            myCamera.startPreview();
            Log.d("Demo", "获取照片成功");
            Toast.makeText(CameraTestActivity.this, "获取照片成功",
                    Toast.LENGTH_SHORT).show();
            /*myCamera.stopPreview();
            myCamera.release();
            myCamera = null;*/
        }
    };
    // 获取文件夹
    private File getDir() {
        // 得到SD卡根目录
        File dir = Environment.getExternalStorageDirectory();
        if (dir.exists()) {
            return dir;
        } else {
            dir.mkdirs();
            return dir;
        }
    }


    @Override
    protected void onDestroy() {
        // TODO Auto-generated method stub
        super.onDestroy();
        if(myCamera != null)
        {
            myCamera.stopPreview();
        myCamera.release();
        myCamera = null;
        }

    }

}
<think>好的,我需要帮助用户在锁屏状态下进入899工模并启动自动测试。首先,我需要理解899工模是什么。通常,工模指的是工程模式,用于设备的硬件测试和调试。不同的设备可能有不同的进入方式,尤其是手机或嵌入式设备。 用户提到在锁屏状态下操作,这可能意味着需要绕过锁屏界面直接进入工模。通常进入工程模式的方式是通过拨号盘输入特定代码,比如*#*#3646633#*#*等,但锁屏状态下可能无法访问拨号盘。因此,可能需要其他方法,比如通过ADB命令或硬件按键组合。 接下来,我需要考虑设备的具体型号和系统。用户提到的引用内容中提到了MTK(联发科)平台的相关信息,比如内置SD卡和恢复模式的操作。这可能意味着设备是基于MTK芯片的,可能需要使用特定的MTK工具或指令。 可能的步骤包括: 1. 使用ADB在锁屏状态下发送指令启动工模应用。需要设备已启用USB调试,并且ADB授权已通过。 2. 使用硬件按键组合,比如音量键和电源键,在启动时进入工程模式。 3. 通过自动化脚本模拟输入,比如使用ADB发送按键事件输入工模代码。 另外,引用[1]提到恢复模式中执行的一些命令,可能涉及到系统分区的操作,但启动自动测试可能需要更直接的触发方式。引用[2]中的设备出现MAC地址提示,可能与蓝牙或网络连接有关,但不确定是否相关。 需要注意的问题包括权限问题,锁屏状态下的安全限制,以及不同设备厂商可能对工模入口做了定制或限制。如果用户设备是特定品牌,可能需要查阅该品牌的工程模式进入方法。 总结可能的解决方案: - 尝试ADB命令启动工模应用:`adb shell am start -n com.mediatek.engineermode/.EngineerMode` - 使用硬件按键组合,如音量下+电源键,在重启时进入工程模式。 - 如果工模有特定的广播Intent,可以通过ADB发送广播触发。 同时,自动测试可能需要执行特定的脚本或命令,这需要工模内部的支持,或者通过ADB连续发送多个指令来完成测试流程。 需要提醒用户注意操作风险,误进入工程模式可能导致设置被更改,影响设备功能。另外,部分操作可能需要root权限,或者厂商定制化的工具支持。</think>要在锁屏状态下进入MTK 899工模并启动自动测试,需结合硬件操作或ADB命令实现。以下是具体步骤和注意事项: --- ### **方法一:通过ADB命令启动工模** 1. **启用USB调试** 需确保设备已提前开启开发者选项中的USB调试功能[^1]。若未开启,需先通过系统设置激活(通常需多次点击“版本号”)。 2. **连接设备并发送指令** 使用ADB工具执行以下命令启动工模应用: ```bash adb shell am start -n com.mediatek.engineermode/.EngineerMode ``` 此命令会直接启动MTK工程模式应用,绕过锁屏界面。 3. **触发自动测试** 进入工模后,通过ADB模拟点击事件或调用特定测试项: ```bash adb shell input tap x y # 根据测试项坐标点击 adb shell input keyevent KEYCODE # 发送按键事件 ``` --- ### **方法二:硬件按键组合** 1. **强制进入工模** 部分MTK设备支持通过按键组合进入工程模式: - 关机状态下,同时按住 **音量下键 + 电源键** - 出现开机LOGO后松开电源键,继续按住音量下键直至进入工模菜单。 2. **导航至自动测试** 使用音量键选择测试模块(如“Auto Test”),按电源键确认启动。 --- ### **注意事项** 1. **权限与兼容性** - ADB方式需设备已授权调试权限,否则无法执行命令[^2]。 - 不同厂商可能定制化修改工模入口路径,需确认包名`com.mediatek.engineermode`是否存在。 2. **风险提示** - 工程模式中的操作可能影响硬件校准或系统稳定性,谨慎修改参数。 - 部分测试项需要特定权限或硬件支持(如射频测试)。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值