如何使用Android opencv使图像的白色部分透明

时间:2016-03-09 08:20:14

标签: android image opencv

我无法将网址链接超过2个,因此我将自己的照片发布到此博客中。 请在这里看看我的问题。 http://blog.naver.com/mail1001/220650041897

我想知道如何制作图像的白色部分,即使用Android opencv透明的白皮书。

我通过网址学习(我在博客上写过),这使得黑色背景变得透明,我认为" Alpha Channel"与它有关。

我认为通过制作我想要制作透明黑色而另一部分为白色的部分制作Alpha通道,并将此Alpha通道合并到原始RGB通道,它将起作用。

所以我做了两次实验。

1)我将papaer部分设为黑色,将写入部分设为白色以制作Alpha通道。并将其合并为RGB Channel。

(请参阅博客。实验1&#39的alpha通道图片)

我认为写作应该是相同的,背景应该是透明的,但背景只会变成白色而且很透明。

(请参阅博客。实验1&#39的结果图片)

2)这次,纸张部分为白色,书写部分为黑色。但这一次只有写作变得透明。

(请参阅博客。实验2&#39的alpha通道图片和结果图片)

在第二个实验中,我的意思是透明变透明但在第一个实验中它的工作方式不同。

我做错了哪一部分?有没有我错误理解的概念?

这是我测试的来源。

Bitmap test(Bitmap image) {
// convert image to matrix
Mat src = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8UC4);
Utils.bitmapToMat(image, src);

// init new matrices
Mat dst = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8UC4);
Mat tmp = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8UC4);
Mat alpha = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8UC4);

// convert image to grayscale
Imgproc.cvtColor(src, tmp, Imgproc.COLOR_BGR2GRAY);

// threshold the image to create alpha channel with complete transparency in black background region and zero transparency in foreground object region.
Imgproc.threshold(tmp, alpha, 100, 255, Imgproc.THRESH_BINARY_INV);
//Imgproc.threshold(tmp, alpha, 100, 255, Imgproc.THRESH_BINARY);

// split the original image into three single channel.
List<Mat> rgb = new ArrayList<Mat>(3);
Core.split(src, rgb);

// Create the final result by merging three single channel and alpha(BGRA order)
List<Mat> rgba = new ArrayList<Mat>(4);
rgba.add(rgb.get(0));
rgba.add(rgb.get(1));
rgba.add(rgb.get(2));
rgba.add(alpha);
Core.merge(rgba, dst);

// convert matrix to output bitmap
Bitmap output = Bitmap.createBitmap(image.getWidth(), image.getHeight(), Bitmap.Config.ARGB_8888);
Utils.matToBitmap(dst, output);
//Utils.matToBitmap(alpha, output);
return output;
}

感谢您的回答。

我尝试了但是它是相同的实验1的结果图。 T_T

编辑代码

Bitmap makeBackgroundWhite(Bitmap image) {
    // convert image to matrix
    Mat src = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8UC4);
    Utils.bitmapToMat(image, src);

    // init new matrices
    Mat dst = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8UC4);
    Mat tmp = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8U);
    Mat alpha = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8U);

    // convert image to grayscale
    Imgproc.cvtColor(src, tmp, Imgproc.COLOR_BGR2GRAY);

    // threshold the image to create alpha channel with complete transparency in black background region and zero transparency in foreground object region.
    Imgproc.threshold(tmp, alpha, 100, 255, Imgproc.THRESH_BINARY_INV);

    // split the original image into three single channel.
    List<Mat> bgra = new ArrayList<Mat>(4);
    Core.split(src, bgra);

    // Create the final result by merging three single channel and alpha(BGRA order)
    bgra.remove(3);
    bgra.add(alpha);
    Core.merge(bgra, dst);

    // convert matrix to output bitmap
    Bitmap output = Bitmap.createBitmap(image.getWidth(), image.getHeight(), Bitmap.Config.ARGB_8888);
    Utils.matToBitmap(dst, output);

    return output;
}

1 个答案:

答案 0 :(得分:3)

您的代码中存在多个问题 我猜两种情况都出错了,但第二种情况只是幸运地假装成功 第二种情况似乎工作正常的原因是字体大多是黑色的 仔细观察心脏(红色)图像,你也可以发现第二个案例也失败了 (你可能不幸运,如果两种情况都失败了你可以立即注意到问题:))

1.您正在使用tmp(灰色)和alpha Mat与CV_8UC4 灰色和alpha通道每个像素只需要1个字节,因此将它们更改为CV_8U。

Mat tmp = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8U);
Mat alpha = new Mat(image.getHeight(), image.getWidth(), CvType.CV_8U);


2.颜色类型令人困惑 使用CV_8UC4初始化src Mat和tmp Mat,并使用转换代码将src转换为tmp:COLOR_BGR2GRAY。
即使实际颜色类型与转换代码不匹配,OpenCV也会转换矩阵,因此您必须同步它们才能有意义。

// suppose you've already changed the type of tmp to CV_8U
Imgproc.cvtColor(src, tmp, Imgproc.COLOR_BGRA2GRAY);


3.拆分和合并
你的src是4通道,你把它分成3个 我不知道OpenCV在这种情况下是如何工作的,但我强烈建议你匹配频道。

ArrayList<Mat> bgra = new ArrayList<Mat>(4);
Core.split(src, bgra);
bgra.remove(3);
bgra.add(alpha);  // suppose your alpha channel is already CV_8U
Core.merge(bgra, dst);



ADDED:C ++版源代码
希望这可以帮助。

// 1. Loading
Mat src = imread("yourImagePath/yourOriginalImage.jpg");  // This code will automatically loads image to Mat with 3-channel(BGR) format

// 2. Grayscaling
Mat gray;
cvtColor(src, gray, CV_BGR2GRAY);   // This will convert BGR src to GRAY

// 3. Thresholding
Mat mask;
threshold(gray, mask, 100, 255, CV_THRES_BINARY); // Or use CV_THRES_BINARY_INV for inverting result

// 4. Splitting & adding Alpha
vector<Mat> channels;   // C++ version of ArrayList<Mat>
split(src, channels);   // Automatically splits channels and adds them to channels. The size of channels = 3
channels.push_back(mask);   // Adds mask(alpha) channel. The size of channels = 4

// 5. Merging
Mat dst;
merge(channels, dst);   // dst is created with 4-channel(BGRA).
// Note that OpenCV applies BGRA by default if your array size is 4,
// even if actual order is different. In this case this makes sense.

// 6. Saving
imwrite("yourImagePath/yourDstImage.png", dst);   // Used PNG format for preserving ALPHA channel