import numpy as np import math from libc.math cimport sqrt cimport numpy as np cimport cython from skimage._shared.geometry cimport point_in_polygon @cython.boundscheck(False) @cython.wraparound(False) def line(int y, int x, int y2, int x2): """Generate line pixel coordinates. Parameters ---------- y, x : int Starting position (row, column). y2, x2 : int End position (row, column). Returns ------- rr, cc : (N,) ndarray of int Indices of pixels that belong to the line. May be used to directly index into an array, e.g. ``img[rr, cc] = 1``. """ cdef np.ndarray[np.int32_t, ndim=1, mode="c"] rr, cc cdef int steep = 0 cdef int dx = abs(x2 - x) cdef int dy = abs(y2 - y) cdef int sx, sy, d, i if (x2 - x) > 0: sx = 1 else: sx = -1 if (y2 - y) > 0: sy = 1 else: sy = -1 if dy > dx: steep = 1 x,y = y,x dx,dy = dy,dx sx,sy = sy,sx d = (2 * dy) - dx rr = np.zeros(int(dx) + 1, dtype=np.int32) cc = np.zeros(int(dx) + 1, dtype=np.int32) for i in range(dx): if steep: rr[i] = x cc[i] = y else: rr[i] = y cc[i] = x while d >= 0: y = y + sy d = d - (2 * dx) x = x + sx d = d + (2 * dy) rr[dx] = y2 cc[dx] = x2 return rr, cc @cython.boundscheck(False) @cython.wraparound(False) @cython.nonecheck(False) def polygon(y, x, shape=None): """Generate coordinates of pixels within polygon. Parameters ---------- y : (N,) ndarray Y-coordinates of vertices of polygon. x : (N,) ndarray X-coordinates of vertices of polygon. shape : tuple, optional image shape which is used to determine maximum extents of output pixel coordinates. This is useful for polygons which exceed the image size. By default the full extents of the polygon are used. Returns ------- rr, cc : ndarray of int Pixel coordinates of polygon. May be used to directly index into an array, e.g. ``img[rr, cc] = 1``. """ cdef int nr_verts = x.shape[0] cdef int minr = max(0, y.min()) cdef int maxr = math.ceil(y.max()) cdef int minc = max(0, x.min()) cdef int maxc = math.ceil(x.max()) # make sure output coordinates do not exceed image size if shape is not None: maxr = min(shape[0]-1, maxr) maxc = min(shape[1]-1, maxc) cdef int r, c #: make contigous arrays for r, c coordinates cdef np.ndarray contiguous_rdata, contiguous_cdata contiguous_rdata = np.ascontiguousarray(y, 'double') contiguous_cdata = np.ascontiguousarray(x, 'double') cdef np.double_t* rptr = contiguous_rdata.data cdef np.double_t* cptr = contiguous_cdata.data #: output coordinate arrays cdef list rr = list() cdef list cc = list() for r in range(minr, maxr+1): for c in range(minc, maxc+1): if point_in_polygon(nr_verts, cptr, rptr, c, r): rr.append(r) cc.append(c) return np.array(rr), np.array(cc) @cython.boundscheck(False) @cython.wraparound(False) @cython.nonecheck(False) @cython.cdivision(True) def ellipse(double cy, double cx, double b, double a, shape=None): """Generate coordinates of pixels within ellipse. Parameters ---------- cy, cx : double Centre coordinate of ellipse. b, a: double Minor and major semi-axes. ``(x/a)**2 + (y/b)**2 = 1``. Returns ------- rr, cc : ndarray of int Pixel coordinates of ellipse. May be used to directly index into an array, e.g. ``img[rr, cc] = 1``. """ cdef int minr = max(0, cy-b) cdef int maxr = math.ceil(cy+b) cdef int minc = max(0, cx-a) cdef int maxc = math.ceil(cx+a) # make sure output coordinates do not exceed image size if shape is not None: maxr = min(shape[0]-1, maxr) maxc = min(shape[1]-1, maxc) cdef int r, c #: output coordinate arrays cdef list rr = list() cdef list cc = list() for r in range(minr, maxr+1): for c in range(minc, maxc+1): if sqrt(((r - cy)/b)**2 + ((c - cx)/a)**2) < 1: rr.append(r) cc.append(c) return np.array(rr), np.array(cc) def circle(double cy, double cx, double radius, shape=None): """Generate coordinates of pixels within circle. Parameters ---------- cy, cx : double Centre coordinate of circle. radius: double Radius of circle. Returns ------- rr, cc : ndarray of int Pixel coordinates of circle. May be used to directly index into an array, e.g. ``img[rr, cc] = 1``. """ return ellipse(cy, cx, radius, radius, shape)