Используя cv2.findContours()
, можно создавать контуры, которые являются «разреженными» (CHAIN_APPROX_SIMPLE) или «полными» (CHAIN_APPROX_NONE). Как я могу преобразовать "редкие" контуры в "полные" контуры?
У меня нет исходного изображения для моих контуров (хотя форма известна), только контуры, и они "разреженные" (CHAIN_APPROX_SIMPLE). Из этого «разреженного» представления я хочу разрешить представление «полное» (CHAIN_APPROX_NONE), чтобы использовать его для доступа к интенсивности контура из других изображений.
Мое временное решение (см. Фрагмент кода) заключается в использовании cv2.drawContours()
, который рисует все пиксели контура из "разреженного" представления контура. Результатом является изображение, и я могу извлечь индексы из полученного изображения, например, используя np.argwhere()
.
Однако, этот дополнительный шаг кажется немного ненужным, учитывая, что cv2.drawContours()
, по-видимому, уже имел эти индексы внутри до рисования получающегося изображения. Я предполагаю, что мне нужен вариант cv2.drawContours()
без чертежной части или опция для вывода «полного» контурного представления вместо изображения.
Другая проблема с моим временным решением заключается в том, что он не сохраняет порядок точек из исходного контура. Интересно, может ли cv2.drawContours()
воссоздать полные действительные контуры внутри, прежде чем сгладить результат в изображение?
Доступна ли эта функция из других функций в opencv, возможно, более базовой функции, используемой внутри cv2.drawContours()
?
import numpy as np
import cv2
# shape (Y,X)
shape = np.array((5, 5))
# sparse contour (X,Y)
sparse_contours = [np.array(
[[[1, 0]],
[[1, 4]],
[[3, 4]],
[[3, 0]]], dtype=np.int32)]
def full_contour_from_contour(shape, contour):
# switch shape from y,x to x,y
shp = shape[[1,0]]
arr = np.zeros(shp, dtype=np.int32)
cv2.drawContours(arr, [contour], 0, 1, 1)
idx = np.argwhere(arr==1)
# reorder Y,X -> X,Y
idx = idx[:, [1, 0]]
# reshape to contour layout
rows, cols = idx.shape
idx = idx.reshape(rows, 1, cols)
return idx.astype(np.int32)
full_contour = full_contour_from_contour(shape, sparse_contour)
# output
# these are correct pixels, with pixels in sparse contour also
# part of the full contour. However, the full contour is not
# necessarily correct or even valid due to
# lost information regarding point sequence along the contour)
[[[1 0]]
[[2 0]]
[[3 0]]
[[1 1]]
[[3 1]]
[[1 2]]
[[3 2]]
[[1 3]]
[[3 3]]
[[1 4]]
[[2 4]]
[[3 4]]]