【问题标题】:How to fill openCV contours with a color specified by its area in Python?如何用 Python 中由其区域指定的颜色填充 openCV 轮廓?
【发布时间】:2020-11-13 19:51:41
【问题描述】:

我已经对生物细胞进行了分割和二进制image,并使用 openCV 提取了轮廓的面积和周长。我正在尝试根据参数 q=perimeter/Sqrt(area) 使用颜色图对每个单元格进行标记和着色,但不知道从哪里开始。根据这个值,基本上每个单元格都会有一个唯一的颜色。

任何帮助将不胜感激!这是我目前所拥有的:

> #specify folders
filelocat = '/Users/kate/Desktop/SegmenterTest3/SegmentedCells/'

#process image
img = cv2.imread(str(filelocat) + 'Seg3.png')
image = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
thresh = cv2.threshold(image, 60, 255, cv2.THRESH_BINARY)[1]
kernel = np.ones((20,20), np.uint8)
closing = cv2.morphologyEx(thresh, cv2.MORPH_CLOSE, kernel)
#inverts image so that the objects are white (for analysis)
imagem = cv2.bitwise_not(closing)

#Find contours
cnts = cv2.findContours(imagem.copy(), cv2.RETR_LIST, cv2.CHAIN_APPROX_SIMPLE)
cnts = imutils.grab_contours(cnts)

#calculate moments and extract cell shape info
moment_dict = {}
for index, cnt in enumerate(cnts):
    moment_dict[index] = cv2.moments(cnt)
    
obj_properties = {}
for index, (key, obj_moments) in enumerate(moment_dict.items()):
    if obj_moments['m00'] > 1000 and obj_moments['m00'] < 20000:
        area = obj_moments['m00']
        cx = obj_moments['m10'] / obj_moments['m00']
        cy = obj_moments['m01'] / obj_moments['m00']
        peri = cv2.arcLength(cnts[index], True)
        q = (peri/(math.sqrt(area)))
        props = {}
        props['q']=q
        props['peri']=peri
        props['area']=area
        props['cx']=cx
        props['cy']=cy
        obj_properties[key] = props

感谢您的帮助!!

【问题讨论】:

  • 颜色可以是随机的还是必须是唯一的颜色列表?
  • 理想情况下,我希望使用颜色图来表示值的渐变,但任何颜色图都可以!
  • 要使用特定的颜色图,您必须拥有与不同区域一样多的定义颜色,在绘制轮廓时必须知道这一点。
  • 在 Google 上搜索 python opencv draw random colors。您将找到诸如stackoverflow.com/questions/28999287/generate-random-colors-rgbstackoverflow.com/questions/19400376/… 之类的答案,用于绘制随机颜色。

标签: python opencv image-processing image-segmentation


【解决方案1】:

要解决这个问题,您需要收集所有 q,以便根据观察到的 q 范围对它们进行缩放。你可以用这样的列表理解来做到这一点:

all_the_q = [v['q'] for k, v in obj_properties.items()]

您还需要选择一些颜色图。根据之前 cmets 中的建议,我将其作为练习留给读者。快速了解一下,您只需将 q 缩放为 8 位 RGB 即可看到初步结果。

请参阅下面的完整代码。请注意,moment_dict 中的 indexobj_properties 字典中的键,因此整个 enumerate 构造是不必要的。我冒昧地完全放弃了enumerate。无论如何,您的过滤循环都会选择正确的轮廓索引。根据您的标准选择轮廓后,收集所有 q 并计算它们的最小/最大/范围。然后使用它们将单个 q 缩放到您需要的任何规模。在下面的示例中,我将其缩放为绿色分量的 8 位值。您可以根据需要遵循红色和蓝色的模式。

请注意,在此图像中,大多数 q 都在 4.0 - 4.25 范围内,在 5.50 处有一些异常值(绘制直方图以查看该分布)。这会扭曲颜色图,因此大多数单元格将被着色为看起来非常相似的颜色。但是,我希望这有助于您入门。我建议对 q 应用对数函数,以便在视觉上“展开”它们分布的下端。

import matplotlib.pyplot as plt
import math
import os
import cv2
import imutils
import numpy as np
# specify folders
filelocat = '/Users/kate/Desktop/SegmenterTest3/SegmentedCells/'

# process image
img = cv2.imread(os.path.join(filelocat, 'Seg3.png'))
image = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
thresh = cv2.threshold(image, 60, 255, cv2.THRESH_BINARY)[1]
kernel = np.ones((20, 20), np.uint8)
closing = cv2.morphologyEx(thresh, cv2.MORPH_CLOSE, kernel)
# inverts image so that the objects are white (for analysis)
imagem = cv2.bitwise_not(closing)

# Find contours
cnts = cv2.findContours(imagem.copy(), cv2.RETR_LIST, cv2.CHAIN_APPROX_SIMPLE)
cnts = imutils.grab_contours(cnts)

# calculate moments and extract cell shape info
moment_dict = {}
for index, cnt in enumerate(cnts):
    moment_dict[index] = cv2.moments(cnt)

obj_properties = {}
for index, obj_moments in moment_dict.items():
    if obj_moments['m00'] > 1000 and obj_moments['m00'] < 20000:
        area = obj_moments['m00']
        cx = obj_moments['m10'] / obj_moments['m00']
        cy = obj_moments['m01'] / obj_moments['m00']
        peri = cv2.arcLength(cnts[index], True)
        q = (peri/(math.sqrt(area)))
        props = {}
        props['q'] = q
        props['peri'] = peri
        props['area'] = area
        props['cx'] = cx
        props['cy'] = cy
        obj_properties[index] = props

all_the_q = [v['q'] for k, v in obj_properties.items()]
min_q = min(all_the_q)
max_q = max(all_the_q)
range_q = max_q - min_q

# colormapping of q scalars to BGR values
cmap = plt.cm.get_cmap('terrain')
for index, prop in obj_properties.items():
    v = (prop['q'] - min_q) / range_q
    r, g, b, a = [int(x) for x in cmap(v, bytes=True)]
    cv2.drawContours(img, cnts, index, (b, g, r), -1)

cv2.imwrite('colored.png', img)
cv2.imshow('Biocells', img)
cv2.waitKey(10000)

【讨论】:

  • 哇,这太棒了!谢谢!我已经设法将它应用到单元格上,并且非常好地标记了它们。现在我试图弄清楚如何去除图像中被切断的外部单元格。祝我好运!
  • 如果您安装了 matplotlib,This 也会很有帮助。您只需要做更多的工作来缩放 RGB 值,解压缩它们,然后按照 BGR 顺序重新排列它们,以便调用 cv2.drawContours
  • 您好,这绝对有效。我仍在研究如何使用对数函数调整配色方案/q 值。但是我遇到了一个新问题。为了移除边界,我添加了一个边框并填充了孔,以消除之前被图像边缘切断的轮廓。现在,当我添加轮廓颜色时,图像会从原始图像偏移,或者当我分析带有边框的新图像时,我会丢失所有颜色。
  • 新代码第 1 部分:#add a border to image bordersize = 10 new_img = cv2.copyMakeBorder(image, top=bordersize, bottom=bordersize, left=bordersize, right=bordersize, borderType=cv2.BORDER_CONSTANT, value=[0, 0, 0]) thresh = cv2.adaptiveThreshold(new_img,255,cv2.ADAPTIVE_THRESH_GAUSSIAN_C, cv2.THRESH_BINARY,11,2) #find contours of the image cnts = cv2.findContours(thresh.copy(), cv2.RETR_LIST, cv2.CHAIN_APPROX_SIMPLE) cnts = imutils.grab_contours(cnts)
  • 新代码第二部分:moment_dict = {} for index, cnt in enumerate(cnts): moment_dict[index] = cv2.moments(cnt) obj_properties = {}
猜你喜欢
  • 2013-04-23
  • 2021-06-23
  • 2013-10-13
  • 1970-01-01
  • 2019-12-22
  • 2022-01-26
  • 2014-11-26
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多