Android 平臺(tái)下基于AnativeWindow實(shí)現(xiàn)相機(jī)預(yù)覽

1、AnativeWindow 官方解釋為:

  • ANativeWindow represents the producer end of an image queue.
  • It is the C counterpart of the android.view.Surface object in Java,
  • and can be converted both ways. Depending on the consumer, images
  • submitted to ANativeWindow can be shown on the display or sent to
  • other consumers, such as video encoders.
    谷歌翻譯:ANativeWindow表示圖像隊(duì)列的生產(chǎn)者端。它是Java中android.view.Surface對(duì)象的C對(duì)應(yīng)物,并且可以雙向轉(zhuǎn)換。 取決于消費(fèi)者,提交到ANativeWindow的圖像可以顯示在顯示器上或發(fā)送給其他消費(fèi)者,例如視頻編碼器。
    因此可以理解的是 Java中可以提供一個(gè)surface通過NDK的方式利用AnativeWindow的API實(shí)現(xiàn)圖像的繪制。
    2、常規(guī)的相機(jī)數(shù)據(jù)處理并顯示的方案:
    ①、利用OpenGL ES 中的shader語(yǔ)言對(duì)圖像進(jìn)行處理;(高效也推薦這個(gè)方法)
    ②、將camera的數(shù)據(jù)傳到底層 進(jìn)行識(shí)別,上層再利用cavans或者其它去繪制我們想表達(dá)的東西,例如:框框、文字等。
    ③、利用AnativeWindow API去實(shí)現(xiàn)相機(jī)圖像數(shù)據(jù)的處理,這種做法存在一定的局限性,對(duì)于一些本身處理比較復(fù)雜的算法,圖像的顯示和繪制會(huì)極為的卡頓。但是AnativeWindow常常用于Video的顯示,算是Android多媒體中一個(gè)比較有用的API。
    3、AnativeWindow顯示流程:
    ①、Camera2 通過ImageReader方式得到相機(jī)data;
    ②、將所得到的Yuvdata通過JNI傳到native層;
    ③、native中將Yuvdata進(jìn)行解碼轉(zhuǎn)成AnativeWindow可以用來繪制的圖像格式;
    AnativeWindow所支持的圖像數(shù)據(jù)格式:
enum ANativeWindow_LegacyFormat {
    // NOTE: these values must match the values from graphics/common/x.x/types.hal

    /** Red: 8 bits, Green: 8 bits, Blue: 8 bits, Alpha: 8 bits. **/
    WINDOW_FORMAT_RGBA_8888          = AHARDWAREBUFFER_FORMAT_R8G8B8A8_UNORM,
    /** Red: 8 bits, Green: 8 bits, Blue: 8 bits, Unused: 8 bits. **/
    WINDOW_FORMAT_RGBX_8888          = AHARDWAREBUFFER_FORMAT_R8G8B8X8_UNORM,
    /** Red: 5 bits, Green: 6 bits, Blue: 5 bits. **/
    WINDOW_FORMAT_RGB_565            = AHARDWAREBUFFER_FORMAT_R5G6B5_UNORM,
};

4、關(guān)鍵代碼
Java:

private ImageReader.OnImageAvailableListener mOnImageAvailableListener = new ImageReader.OnImageAvailableListener() {

        /*
         *  The following method will be called every time an image is ready
         *  be sure to use method acquireNextImage() and then close(), otherwise, the display may STOP
         */
        @Override
        public void onImageAvailable(ImageReader reader) {
            // get the newest frame
            Image image = reader.acquireNextImage();

            if (image == null) {
                return;
            }
            final byte[] yuvdata=ImageUtil.getBytesFromImageAsType(image,0);
            draw(image.getWidth(),image.getHeight(),yuvdata,surface);
            Log.d("pctest","yuvdata length:"+yuvdata.length+"image height:"+image.getHeight()+"image width"+image.getWidth());
            image.close();
        }
    };

private void startPreview(CameraDevice mCamera) throws CameraAccessException {
        SurfaceTexture texture = mPreviewView.getSurfaceTexture();
        texture.setDefaultBufferSize(mPreviewSize.getWidth(),mPreviewSize.getHeight());
        surface = new Surface(texture);
        try {
            // to set request for PREVIEW
            mPreviewBuilder = mCamera.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW);
        } catch (CameraAccessException e) {
            e.printStackTrace();
        }
        mImageReader = ImageReader.newInstance(mImageWidth,mImageHeight, ImageFormat.YUV_420_888,2);
        mImageReader.setOnImageAvailableListener(mOnImageAvailableListener,mHandler);
        mPreviewBuilder.addTarget(mImageReader.getSurface());
        List<Surface> outputSurfaces = new ArrayList<>();
        outputSurfaces.add(mImageReader.getSurface());
        mCamera.createCaptureSession(outputSurfaces, mSessionStateCallback, mHandler);
    }

Native:
在對(duì)數(shù)據(jù)處理中利用的 OpenCV 進(jìn)行圖像數(shù)據(jù)轉(zhuǎn)碼,只是為了方便= 。=(PS:轉(zhuǎn)碼這是太。。。)

extern "C"
JNIEXPORT void JNICALL
Java_com_example_anativewindow4camera2_MainActivity_draw(JNIEnv *env, jobject thiz, jint width,
                                                         jint height, jbyteArray yuvdata,
                                                         jobject surface) {
    // TODO: implement draw()
    jbyte *data = env->GetByteArrayElements(yuvdata, NULL);
    Mat src(height + height / 2, width, CV_8UC1, data);
    cvtColor(src, src, COLOR_YUV2RGBA_IYUV);
    cv::transpose(src,src);
    cv::flip(src,src,1);
    ANativeWindow * window = ANativeWindow_fromSurface(env, surface);
    ANativeWindow_acquire(window);
    ANativeWindow_Buffer buffer;
    ANativeWindow_setBuffersGeometry(window, src.cols, src.rows, WINDOW_FORMAT_RGBA_8888);
    if (int32_t err = ANativeWindow_lock(window, &buffer, NULL)) {
        LOGE("ANativeWindow_lock failed with error code: %d\n", err);
        ANativeWindow_release(window);
    }
    uint8_t * outPtr = reinterpret_cast<uint8_t *>(buffer.bits);
    int dst_line_size = buffer.stride * 4;
    //一行一行拷貝
    for (int i = 0; i < buffer.height; ++i) {
        //void *memcpy(void *dest, const void *src, size_t n);
        //從源src所指的內(nèi)存地址的起始位置開始拷貝n個(gè)字節(jié)到目標(biāo)dest所指的內(nèi)存地址的起始位置中
        memcpy(outPtr + i * dst_line_size,
               src.data + i * src.cols * 4, dst_line_size);
    }

    ANativeWindow_unlockAndPost(window);
    ANativeWindow_release(window);
    src.release();
    env->ReleaseByteArrayElements(yuvdata, data, 0);
}

5、記錄一下,怕后面忘了。。。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

友情鏈接更多精彩內(nèi)容