我创建了一个绘画应用程序,在其中允许用户绘制并保存图像以便以后重新加载继续绘制。基本上,我将绘图作为位图传递到JNI层进行保存,并使用相同的方法来加载先前的绘图。
经过进一步的研究,我发现如果我直接从位图获取int像素数组并将其直接传递给JNI,则不会出现此问题,而不是当前所做的将整个位图传递到JNI层,然后获取像素并使用
我正在使用OpenCv来写入和读取png文件。
在图像的透明度方面,我注意到一些奇怪的地方。似乎透明度是针对OpenCv中的黑色计算出来的?请看附带的图片,它们包含具有透明度的线条。
通过将int数组传递给本地代码来正确处理透明度,不需要进行颜色转换:
通过将Bitmap对象传递给本地代码来加深透明度,需要进行颜色转换:
可能会发生什么问题?
使用本机Bitmap get pixel方法保存图像:
if ((error = AndroidBitmap_getInfo(pEnv, jbitmap, &info)) < 0) {
LOGE("AndroidBitmap_getInfo() failed! error:%d",error);
}
if (0 == error)
{
if ((error = AndroidBitmap_lockPixels(pEnv, jbitmap, &pixels)) < 0) {
LOGE("AndroidBitmap_lockPixels() failed ! error=%d", error);
}
}
if (0 == error)
{
if (info.format == ANDROID_BITMAP_FORMAT_RGBA_8888)
{
LOGI("ANDROID_BITMAP_FORMAT_RGBA_8888");
}
else
{
LOGI("ANDROID_BITMAP_FORMAT %d",info.format);
}
Mat bgra(info.height, info.width, CV_8UC4, pixels);
Mat image;
//bgra.copyTo(image);
// fix pixel order RGBA -> BGRA
cvtColor(bgra, image, COLOR_RGBA2BGRA);
vector<int> compression_params;
compression_params.push_back(CV_IMWRITE_PNG_COMPRESSION);
compression_params.push_back(3);
// save image
if (!imwrite(filePath, image, compression_params))
{
LOGE("saveImage() -> Error saving image!");
error = -7;
}
// release locked pixels
AndroidBitmap_unlockPixels(pEnv, jbitmap);
}
使用本地int像素数组方法保存图像:
JNIEXPORT void JNICALL Java_com_vblast_smasher_Smasher_saveImageRaw
(JNIEnv *pEnv, jobject obj, jstring jFilePath, jintArray jbgra, jint options, jint compression)
{
jint* _bgra = pEnv->GetIntArrayElements(jbgra, 0);
const char *filePath = pEnv->GetStringUTFChars(jFilePath, 0);
if (NULL != filePath)
{
Mat image;
Mat bgra(outputHeight, outputWidth, CV_8UC4, (unsigned char *)_bgra);
bgra.copyTo(image);
if (0 == options)
{
// replace existing cache value
mpCache->insert(filePath, image);
}
vector<int> compression_params;
compression_params.push_back(CV_IMWRITE_PNG_COMPRESSION);
compression_params.push_back(compression);
// save image
if (!imwrite(filePath, image))
{
LOGE("saveImage() -> Error saving image!");
}
}
pEnv->ReleaseIntArrayElements(jbgra, _bgra, 0);
pEnv->ReleaseStringUTFChars(jFilePath, filePath);
}
更新 05/25/12:经过进一步的研究,我发现如果我直接从位图获取int像素数组并将其直接传递给JNI,则不会出现此问题,而不是当前所做的将整个位图传递到JNI层,然后获取像素并使用
cvtColor
正确转换像素。 我是否使用了正确的像素转换?
compression
的值是多少? - karlphillip