python - 将图像环绕一个圆圈

标签 python python-3.x python-imaging-library

在这个例子中,我想做的是将图像环绕成一个圆圈,如下所示。

original

wrapped

为了包裹图像,我简单地使用三角函数计算了 x,y 坐标。 问题是计算出的 X 和 Y 位置四舍五入使它们成为整数。这会导致在上面的包裹图像中看到空白像素。 x,y 位置必须是整数,因为它们是列表中的位置。

我在下面的代码中再次执行了此操作,但没有任何图像以便于查看。我所做的就是创建两个具有二进制值的数组,一个数组是黑色的,另一个是白色的,然后将一个数组包裹到另一个数组上。

代码的输出是。

example

import math as m
from PIL import Image  # only used for showing output as image

width = 254.0
height = 24.0
Ro = 40.0

img = [[1 for x in range(int(width))] for y in range(int(height))]
cir = [[0 for x in range(int(Ro * 2))] for y in range(int(Ro * 2))]


def shom_im(img):  # for showing data as image
    list_image = [item for sublist in img for item in sublist]
    new_image = Image.new("1", (len(img[0]), len(img)))
    new_image.putdata(list_image)
    new_image.show()

increment = m.radians(360 / width)
rad = Ro - 0.5
for i, row in enumerate(img):
    hyp = rad - i
    for j, column in enumerate(row):
        alpha = j * increment
        x = m.cos(alpha) * hyp + rad
        y = m.sin(alpha) * hyp + rad
        # put value from original image to its position in new image
        cir[int(round(y))][int(round(x))] = img[i][j]


shom_im(cir)

我后来发现了中点圆算法,但结果更差

midpoint

from PIL import Image  # only used for showing output as image

width, height = 254, 24
ro = 40

img = [[(0, 0, 0, 1) for x in range(int(width))]
       for y in range(int(height))]
cir = [[(0, 0, 0, 255) for x in range(int(ro * 2))] for y in range(int(ro * 2))]


def shom_im(img):  # for showing data as image
    list_image = [item for sublist in img for item in sublist]
    new_image = Image.new("RGBA", (len(img[0]), len(img)))
    new_image.putdata(list_image)
    new_image.show()


def putpixel(x0, y0):
    global cir
    cir[y0][x0] = (255, 255, 255, 255)


def drawcircle(x0, y0, radius):
    x = radius
    y = 0
    err = 0

    while (x >= y):
        putpixel(x0 + x, y0 + y)
        putpixel(x0 + y, y0 + x)
        putpixel(x0 - y, y0 + x)
        putpixel(x0 - x, y0 + y)
        putpixel(x0 - x, y0 - y)
        putpixel(x0 - y, y0 - x)
        putpixel(x0 + y, y0 - x)
        putpixel(x0 + x, y0 - y)
        y += 1
        err += 1 + 2 * y
        if (2 * (err - x) + 1 > 0):
            x -= 1
            err += 1 - 2 * x

for i, row in enumerate(img):
    rad = ro - i
    drawcircle(int(ro - 1), int(ro - 1), rad)

shom_im(cir)

谁能建议一种消除空白像素的方法?

最佳答案

你在填满你的圈子时遇到了问题,因为你是从错误的方式来处理这个问题的——毫不夸张地说。

当将源映射到目标时,您需要填充目标,并将每个已翻译的像素从该源映射到< em>来源 图片。这样一来,您就绝不会错过一个像素,同样,您永远不会多次绘制(或查找)一个像素。

以下内容比较粗略,仅作为概念示例。我首先编写了一些代码来绘制一个从上到下的实心圆。然后我添加了更多代码以删除中心部分(并添加了一个变量 Ri,用于“内半径”)。这会形成一个实心环,其中所有像素仅绘制一次:从上到下,从左到右。

a solid ring

你究竟如何绘制环其实并不重要!我最初使用三角是因为我想重新使用角度钻头,但它也可以用 Pythagorus 来完成,甚至可以用 Bresenham 的圆例程来完成。您需要牢记的是,您迭代的是目标 行和列,而不是。这提供了实际的 x,y 坐标,您可以将其输入到重新映射过程中。

完成上述工作后,我编写了三角函数以将 坐标转换为 原始图像中的像素。为此,我创建了一个包含一些文本的测试图像:

test image with text

这也是一件好事,因为在第一次尝试中我得到了两次文本(一次向左,一次向右)并进行了镜像——这需要一些小的调整。还要注意背景网格。我添加它是为了检查“顶部”和“底部”线(最外圈和最内圈)是否正确绘制。

使用此图像和 RoRi 在 100 和 50 运行我的代码,我得到以下结果:

round test image

您可以看到三角函数使它从最右边的点开始,顺时针移动,并使图像的顶部指向外。所有这些都可以微调,但这样它会模仿您想要绘制图像的方向。

这是你的虹膜图像的结果,使用 33 作为内半径:

iris to eye mapping

这是一个很好的动画,展示了映射的稳定性:

aww bright light ahead!

最后,那么,我的代码是:

import math as m
from PIL import Image

Ro = 100.0
Ri = 50.0

# img = [[1 for x in range(int(width))] for y in range(int(height))]
cir = [[0 for x in range(int(Ro * 2))] for y in range(int(Ro * 2))]

# image = Image.open('0vWEI.png')
image = Image.open('this-is-a-test.png')
# data = image.convert('RGB')
pixels = image.load()
width, height = image.size

def shom_im(img):  # for showing data as image
    list_image = [item for sublist in img for item in sublist]
    new_image = Image.new("RGB", (len(img[0]), len(img)))
    new_image.putdata(list_image)
    new_image.save("result1.png","PNG")
    new_image.show()


for i in range(int(Ro)):
    # outer_radius = Ro*m.cos(m.asin(i/Ro))
    outer_radius = m.sqrt(Ro*Ro - i*i)
    for j in range(-int(outer_radius),int(outer_radius)):
        if i < Ri:
            # inner_radius = Ri*m.cos(m.asin(i/Ri))
            inner_radius = m.sqrt(Ri*Ri - i*i)
        else:
            inner_radius = -1
        if j < -inner_radius or j > inner_radius:
            # this is the destination
            # solid:
            # cir[int(Ro-i)][int(Ro+j)] = (255,255,255)
            # cir[int(Ro+i)][int(Ro+j)] = (255,255,255)
            # textured:

            x = Ro+j
            y = Ro-i
            # calculate source
            angle = m.atan2(y-Ro,x-Ro)/2
            distance = m.sqrt((y-Ro)*(y-Ro) + (x-Ro)*(x-Ro))
            distance = m.floor((distance-Ri+1)*(height-1)/(Ro-Ri))
        #   if distance >= height:
        #       distance = height-1
            cir[int(y)][int(x)] = pixels[int(width*angle/m.pi) % width, height-distance-1]
            y = Ro+i
            # calculate source
            angle = m.atan2(y-Ro,x-Ro)/2
            distance = m.sqrt((y-Ro)*(y-Ro) + (x-Ro)*(x-Ro))
            distance = m.floor((distance-Ri+1)*(height-1)/(Ro-Ri))
        #   if distance >= height:
        #       distance = height-1
            cir[int(y)][int(x)] = pixels[int(width*angle/m.pi) % width, height-distance-1]

shom_im(cir)

注释掉的线条绘制了一个实心的白色圆环。注意这里和那里的各种调整以获得最佳结果。例如,distance 是从圆心开始测量的,因此接近圆心时返回低值,圆外时返回最大值。将其直接映射回目标图像将显示文本,其顶部“向内”指向内孔。所以我用 height - distance - 1 反转这个映射,其中 -1 是让它从 0 映射到 height 再次。

distance 本身的计算也有类似的修复;没有调整 Ri+1height-1 最里面或最外面的行都不会被绘制,表明计算只差一个像素(这正是该网格的目的)。

关于python - 将图像环绕一个圆圈,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38745020/

相关文章:

使用 API 获取 json 中的 xml 数据的 python 脚本

python - 在 python3 中用大列表总结旅行时间

python - 使用 PIL 在 App Engine 上绘制图像

python - 如何使用 Python 为 Azure 进行原始 REST 调用

python - distutils.core.setup 控制台脚本入口点?

python - 将 Numpy 数组转换为 Pandas DataFrame

python - 用红色或黑色像素替换2D矩阵值

python - 从 numpy 数组中删除连续的 RGB 值

python - 查找具有多个 'child' 对象的 Django 模型记录?

python - 创建新列表初始化