7

基本上我想为我的油漆应用程序实现颜色替换功能。以下是原始和预期的输出

原来的: 原来的

在更改用户选择的墙壁颜色以及一些替换阈值后

期望的输出 我尝试了两种方法,但无法按预期工作

方法 1:用于颜色替换
的基于队列的 Flood Fill算法,但我的输出速度非常慢,并且没有保留墙阴影。

洪水填充输出

方法 2: 所以我试图查看另一个选项,并在下面的帖子中找到了 SO 如何更改图像中的特定颜色?

但我无法理解逻辑,也不确定我在第 3 步中的代码实现。

请根据我的理解为每一步找到以下代码。

1) 使用 cvCvtColor 将图像从 RGB 转换为 HSV(我们只想改变色调)。

 IplImage *mainImage=[self CreateIplImageFromUIImage:[UIImage imageNamed:@"original.jpg"]];
 IplImage *hsvImage = cvCreateImage(cvGetSize(mainImage), IPL_DEPTH_8U, 3);
 IplImage *threshImage = cvCreateImage(cvGetSize(mainImage), IPL_DEPTH_8U, 3);
 cvCvtColor(mainImage,hsvImage,CV_RGB2HSV);

2) 用 cvThreshold 指定一定的容差来隔离颜色(您需要一系列颜色,而不是单一颜色)。

cvThreshold(hsvImage, threshImage, 0, 100, CV_THRESH_BINARY);

3) 使用 cvBlobsLib 之类的斑点检测库丢弃颜色低于最小尺寸的区域。这将消除场景中相似颜色的点。我需要指定原始图像或阈值图像吗?

CBlobResult blobs = CBlobResult(threshImage, NULL, 0);
blobs.Filter( blobs, B_EXCLUDE, CBlobGetArea(), B_LESS, 10);

4) 用 cvInRangeS 遮罩颜色,并使用生成的遮罩应用新的色调。

不确定此功能如何帮助颜色替换并且无法理解要提供的参数。

5) cvMerge 新图像与新色调与由您在第一步中保存的饱和度和亮度通道组成的图像。

我知道 cvMerge 将合并 HS 和 V 的三个通道,但是我如何使用上述三个步骤的输出。

所以基本上坚持使用opencv实现,

如果可能的话,请指导我进行 opencv 实施或任何其他尝试解决方案。

4

2 回答 2

4

最后,我能够使用下面的 javacv 代码实现一些所需的输出,并且也可以移植到 opencv。

这个解决方案有两个问题

  1. 没有边缘检测,我认为使用轮廓我可以实现它
  2. 替换的颜色具有平坦的色调和饱和度,应根据源像素色调饱和度差异进行设置,但不确定如何实现。可以使用 cvAddS 代替 cvSet

    IplImage image = cvLoadImage("sample.png");
    CvSize cvSize = cvGetSize(image);
    
    
    IplImage hsvImage = cvCreateImage(cvSize, image.depth(),image.nChannels());
    
    IplImage hChannel = cvCreateImage(cvSize, image.depth(), 1); 
            IplImage  sChannel = cvCreateImage(cvSize, image.depth(), 1); 
            IplImage  vChannel = cvCreateImage(cvSize, image.depth(), 1);
    cvSplit(hsvImage, hChannel, sChannel, vChannel, null);
    
    
    IplImage cvInRange = cvCreateImage(cvSize, image.depth(), 1);
    CvScalar source=new CvScalar(72/2,0.07*255,66,0); //source color to replace
    CvScalar from=getScaler(source,false);
    CvScalar to=getScaler(source, true);
    
    cvInRangeS(hsvImage, from , to, cvInRange);
    
    IplImage dest = cvCreateImage(cvSize, image.depth(), image.nChannels());
    
    IplImage temp = cvCreateImage(cvSize, IPL_DEPTH_8U, 2);
    cvMerge(hChannel, sChannel, null, null, temp);
    
    cvSet(temp, new CvScalar(45,255,0,0), cvInRange);// destination hue and sat
    cvSplit(temp, hChannel, sChannel, null, null);
    cvMerge(hChannel, sChannel, vChannel, null, dest);
    cvCvtColor(dest, dest, CV_HSV2BGR);
    cvSaveImage("output.png", dest);
    

阈值计算方法

    CvScalar getScaler(CvScalar seed,boolean plus){
    if(plus){
        return CV_RGB(seed.red()+(seed.red()*thresold),seed.green()+(seed.green()*thresold),seed.blue()+(seed.blue()*thresold));
    }else{
        return CV_RGB(seed.red()-(seed.red()*thresold),seed.green()-(seed.green()*thresold),seed.blue()-(seed.blue()*thresold));
    }
        }
于 2013-03-01T04:47:53.257 回答
0

我知道这个答案有一天会对某人有用。在 iOS 的 viewdidLoad() 覆盖方法中试试这个。下面代码片段中的图像应该来自您的 UIImageView 种子也是固定的。您可以根据用户点击事件使其动态化。

 cv::Mat mask = cv::Mat::zeros(image.rows + 2, image.cols + 2, CV_8U);
 imageView.image = [self UIImageFromCVMat:image];
 cv::cvtColor(image, image, cv::COLOR_BGR2RGB);

  try {
    if(seed.x > 0 && seed.y > 0){
        cv::floodFill(image, mask, seed, cv::Scalar(50, 155, 20) ,0,        cv::Scalar(2,2, 2), cv::Scalar(2,2, 2), 8);
        cv::floodFill(image, mask, seed2, cv::Scalar(50, 155, 20) ,0, cv::Scalar(2,2, 2), cv::Scalar(2,2, 2), 8);
        cv::floodFill(image, mask, seed3, cv::Scalar(50, 155, 0) ,0, cv::Scalar(2,2, 2), cv::Scalar(2,2, 2), 8);

    }
} catch (Exception ex) {

}
cv::cvtColor(image, image, cv::COLOR_RGB2BGR);

self.imageView.contentMode = UIViewContentModeScaleAspectFill;
self.imageView.image = [self UIImageFromCVMat:image];
于 2018-09-06T18:16:11.907 回答