android全平台编译libpng并基于ANativeWindow加载PNG图片

图形图像实践

环境配置

操作系统:ubuntu 16.05
ndk版本:android-ndk-r16b
libpng版本:libpng-1.6.35
zlib版本:zlib-1.2.11

进入libpng-1.6.35/scripts目录,将pnglibconf.h.prebuilt复制到libpng-1.6.35目录下,并重命名为pnglibconf.h

开始执行ndk-build

加载PNG图片

新建native-png工程

新建NativePngLoader类

package com.onzhou.graphic.png;

import android.view.Surface;

public class NativePngLoader {

    static {
        System.loadLibrary("native-png");
    }

    public native void loadPNGImage(String imagePath, Surface surface);


}

新建native_png.cpp的native实现类

#include <jni.h>
#include <stdio.h>
#include <time.h>
#include <android/native_window.h>
#include <android/native_window_jni.h>

#include "native_png.h"
#include "png.h"
#include "zlib.h"
#include <malloc.h>
#include <string.h>

#ifdef ANDROID

#include <android/log.h>


#define LOG_TAG    "NativePNG"
#define LOGE(format, ...)  __android_log_print(ANDROID_LOG_ERROR, LOG_TAG, format, ##__VA_ARGS__)
#define LOGI(format, ...)  __android_log_print(ANDROID_LOG_INFO,  LOG_TAG, format, ##__VA_ARGS__)
#else
#define LOGE(format, ...)  printf(LOG_TAG format "\n", ##__VA_ARGS__)
#define LOGI(format, ...)  printf(LOG_TAG format "\n", ##__VA_ARGS__)
#endif


/**
 * 动态注册
 */
JNINativeMethod methods[] = {
        {"loadPNGImage", "(Ljava/lang/String;Landroid/view/Surface;)V", (void *) loadPNGImage}
};

/**
 * 动态注册
 * @param env
 * @return
 */
jint registerNativeMethod(JNIEnv *env) {
    jclass cl = env->FindClass("com/onzhou/graphic/png/NativePngLoader");
    if ((env->RegisterNatives(cl, methods, sizeof(methods) / sizeof(methods[0]))) < 0) {
        return -1;
    }
    return 0;
}

/**
 * 加载默认回调
 * @param vm
 * @param reserved
 * @return
 */
jint JNI_OnLoad(JavaVM *vm, void *reserved) {
    JNIEnv *env = NULL;
    if (vm->GetEnv((void **) &env, JNI_VERSION_1_6) != JNI_OK) {
        return -1;
    }
    //注册方法
    if (registerNativeMethod(env) != JNI_OK) {
        return -1;
    }
    return JNI_VERSION_1_6;
}

void ThrowException(JNIEnv *env, const char *exception, const char *message) {
    jclass clazz = env->FindClass(exception);
    if (NULL != clazz) {
        env->ThrowNew(clazz, message);
    }
}

void drawPNG(const char *name, ANativeWindow_Buffer &nwBuffer) {
    FILE *file = fopen(name, "rb");

    png_structp png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, 0, 0, 0);
    if (png_ptr == NULL) {
        fclose(file);
        return;
    }
    png_infop info_ptr = png_create_info_struct(png_ptr);
    if (info_ptr == NULL) {
        fclose(file);
        png_destroy_write_struct(&png_ptr, NULL);
        return;
    }
    if (setjmp(png_jmpbuf(png_ptr))) {
        fclose(file);
        png_destroy_write_struct(&png_ptr, &info_ptr);
        return;
    }
    //开始读文件
    png_init_io(png_ptr, file);

    png_read_png(png_ptr, info_ptr, PNG_TRANSFORM_EXPAND, 0);

    //获取文件的宽高色深
    int m_width = png_get_image_width(png_ptr, info_ptr);
    int m_height = png_get_image_height(png_ptr, info_ptr);

    //获取图像的色彩类型
    int color_type = png_get_color_type(png_ptr, info_ptr);

    LOGI("width=%d,height=%d,color_type=%d", m_width, m_height, color_type);

    // row_pointers内部存放的就是RGBA数据了
    png_bytep *row_pointers = png_get_rows(png_ptr, info_ptr);

    switch (color_type) {
        case PNG_COLOR_TYPE_RGB_ALPHA:
            //RGBA32
            break;
        case PNG_COLOR_TYPE_RGB: {
            //RGB24,没有A通道,就要3位3位的读
            uint32_t *line = (uint32_t *) nwBuffer.bits;
            for (int row = 0; row < m_height; row++) {
                for (int column = 0; column < m_width; column++) {
                    //存储顺序为BGR,BGR,BGR......
                    line[column] = ((uint32_t) row_pointers[row][3 * column + 2]) << 16
                                   | ((uint32_t) row_pointers[row][3 * column + 1]) << 8
                                   | (uint32_t) row_pointers[row][3 * column];
                }
                line = line + nwBuffer.stride;
            }
        }
    }
    png_destroy_read_struct(&png_ptr, &info_ptr, 0);
    fclose(file);
}


void loadPNGImage(JNIEnv *env, jobject obj, jstring jpegPath, jobject surface) {

    const char *path = env->GetStringUTFChars(jpegPath, 0);
    //获取目标surface
    ANativeWindow *window = ANativeWindow_fromSurface(env, surface);
    if (NULL == window) {
        ThrowException(env, "java/lang/RuntimeException", "unable to get native window");
        return;
    }
    //默认的是RGB_565
    int32_t result = ANativeWindow_setBuffersGeometry(window, 0, 0, WINDOW_FORMAT_RGBA_8888);
    if (result < 0) {
        ThrowException(env, "java/lang/RuntimeException", "unable to set buffers geometry");
        //释放窗口
        ANativeWindow_release(window);
        window = NULL;
        return;
    }
    ANativeWindow_acquire(window);

    ANativeWindow_Buffer buffer;
    //锁定窗口的绘图表面
    if (ANativeWindow_lock(window, &buffer, NULL) < 0) {
        ThrowException(env, "java/lang/RuntimeException", "unable to lock native window");
        //释放窗口
        ANativeWindow_release(window);
        window = NULL;
        return;
    }

    //绘制PNG图片
    drawPNG(path, buffer);

    //解锁窗口的绘图表面
    if (ANativeWindow_unlockAndPost(window) < 0) {
        ThrowException(env, "java/lang/RuntimeException",
                       "unable to unlock and post to native window");
    }

    env->ReleaseStringUTFChars(jpegPath, path);
    //释放
    ANativeWindow_release(window);
}

注意:上面的过程其实也比较简单,但是跟读取JPEG图片区别在于要根据色彩来区分RGBRGBA

  • 先通过ANativeWindow_fromSurface获取对应的窗口
  • 通过libpng加载对应的png图片,解码成RGBA数据
  • 将最终的RGBA数据写入到buffer中去,完成绘制

编写cmake的配置文件CMakeLists.txt


cmake_minimum_required(VERSION 3.4.1)

##官方标准配置
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wall")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11 -fno-rtti -fno-exceptions -Wall")

add_library(native-png
           SHARED
           src/main/cpp/native_png.cpp)

add_library(png
           SHARED
           IMPORTED)

set_target_properties(png
                    PROPERTIES IMPORTED_LOCATION
                    ${CMAKE_SOURCE_DIR}/libs/${ANDROID_ABI}/libpng.so
                    )

add_library(zip
           SHARED
           IMPORTED)

set_target_properties(zip
                    PROPERTIES IMPORTED_LOCATION
                    ${CMAKE_SOURCE_DIR}/libs/${ANDROID_ABI}/libz.so
                    )

#头文件
include_directories(${CMAKE_SOURCE_DIR}/src/main/cpp/include_png)
include_directories(${CMAKE_SOURCE_DIR}/src/main/cpp/include_zlib)

target_link_libraries(native-png
            png
            zip
            android
            jnigraphics
            log)

在启动的目标Activity中加载我们指定的JPEG图片

mSurfaceView.getHolder().addCallback(new SurfaceHolder.Callback() {
    @Override
    public void surfaceCreated(SurfaceHolder holder) {
        NativeImageLoader nativeImageLoader = new NativeImageLoader();
        File file = new File(getExternalFilesDir(null), "input.jpeg");
        nativeImageLoader.loadJPEGImage(file.getAbsolutePath(), holder.getSurface());
    }

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

    }

    @Override
    public void surfaceDestroyed(SurfaceHolder holder) {

    }
});

可以看到如下输出:

项目地址:native-png
https://github.com/byhook/graphic4android

参考:
https://blog.csdn.net/talkxin/article/details/50968313

基于libpng库将PNG图片数据转换为RGB数据的过程涉及到几个主要步骤。libpng是一个用于读取和写入PNG(便携式网络图形)格式图片的库。以下是使用libpng库将PNG图片转换为RGB数据的大致步骤: 1. 初始化libpng库,设置必要的函数指针。 2. 打开PNG图片文件,并创建一个png_structp结构体用于存储库的读取状态。 3. 创建一个png_infop结构体用于存储读取PNG图片时的信息。 4. 使用libpng的读取函数加载PNG图片,并对PNG文件头信息进行处理。 5. 读取PNG图片的像素数据。PNG图片可能包含多种类型的像素格式,比如索引彩色、灰度、RGB等。确保解码为RGB格式。 6. 将PNG图片像素数据转换为RGB格式,通常每像素3个字节,分别代表红色、绿色和蓝色通道。 7. 清理libpng使用过的资源,关闭PNG文件,并释放相关结构体。 下面是一个简化的代码示例,展示了如何使用libpng库读取PNG文件并提取RGB数据: ```c #include <png.h> #include <stdio.h> int main(int argc, char *argv[]) { FILE *fp = fopen("example.png", "rb"); png_structp png_ptr = NULL; png_infop info_ptr = NULL; png_bytep row = NULL; int width, height; // 初始化libpng结构体 if ((png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL)) == NULL) return -1; if ((info_ptr = png_create_info_struct(png_ptr)) == NULL) { png_destroy_read_struct(&png_ptr, NULL, NULL); return -1; } // 设置错误处理函数 if (setjmp(png_jmpbuf(png_ptr))) { png_destroy_read_struct(&png_ptr, &info_ptr, NULL); if (row) free(row); fclose(fp); return -1; } // 设置libpng输入流 png_init_io(png_ptr, fp); // 读取PNG文件头信息 png_read_info(png_ptr, info_ptr); width = png_get_image_width(png_ptr, info_ptr); height = png_get_image_height(png_ptr, info_ptr); // 分配内存来存储每一行数据 row = (png_bytep) malloc(3 * width * sizeof(png_byte)); // 读取每一行数据 for (int y = 0; y < height; y++) { png_read_row(png_ptr, row, NULL); // 此处处理row中的数据,将其转换为RGB数据 } // 清理工作 if (row) free(row); png_destroy_read_struct(&png_ptr, &info_ptr, NULL); fclose(fp); return 0; } ``` 请注意,这只是一个非常基础的示例。在实际的应用中,你可能需要处理各种PNG图片的不同特性,比如透明度(alpha通道)、调色板(palette-based images)和不同的像素深度。同时,错误处理和资源管理需要更加严谨。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值