python 如何在图像周围添加圆形边框?

brgchamk  于 2024-01-05  发布在  Python
关注(0)|答案(5)|浏览(213)

我有一个矩形图像,我想圆它的角落,然后添加一个黑色的边界(所以边界也是圆的)。
有没有一个简单的方法来实现它?
这将是所需的输出:


的数据
Similar unanswered question

nxowjjhe

nxowjjhe1#

在我的第一个答案的评论中与Mark进行了一些讨论后,我决定使用OpenCV和NumPy来制作另一个解决方案,它能够轻松地将一些真实的图像(例如照片)提供给该方法,并获得包括圆角边框和边框外透明度的图像!

  1. import cv2
  2. import numpy as np
  3. def rect_with_rounded_corners(image, r, t, c):
  4. """
  5. :param image: image as NumPy array
  6. :param r: radius of rounded corners
  7. :param t: thickness of border
  8. :param c: color of border
  9. :return: new image as NumPy array with rounded corners
  10. """
  11. c += (255, )
  12. h, w = image.shape[:2]
  13. # Create new image (three-channel hardcoded here...)
  14. new_image = np.ones((h+2*t, w+2*t, 4), np.uint8) * 255
  15. new_image[:, :, 3] = 0
  16. # Draw four rounded corners
  17. new_image = cv2.ellipse(new_image, (int(r+t/2), int(r+t/2)), (r, r), 180, 0, 90, c, t)
  18. new_image = cv2.ellipse(new_image, (int(w-r+3*t/2-1), int(r+t/2)), (r, r), 270, 0, 90, c, t)
  19. new_image = cv2.ellipse(new_image, (int(r+t/2), int(h-r+3*t/2-1)), (r, r), 90, 0, 90, c, t)
  20. new_image = cv2.ellipse(new_image, (int(w-r+3*t/2-1), int(h-r+3*t/2-1)), (r, r), 0, 0, 90, c, t)
  21. # Draw four edges
  22. new_image = cv2.line(new_image, (int(r+t/2), int(t/2)), (int(w-r+3*t/2-1), int(t/2)), c, t)
  23. new_image = cv2.line(new_image, (int(t/2), int(r+t/2)), (int(t/2), int(h-r+3*t/2)), c, t)
  24. new_image = cv2.line(new_image, (int(r+t/2), int(h+3*t/2)), (int(w-r+3*t/2-1), int(h+3*t/2)), c, t)
  25. new_image = cv2.line(new_image, (int(w+3*t/2), int(r+t/2)), (int(w+3*t/2), int(h-r+3*t/2)), c, t)
  26. # Generate masks for proper blending
  27. mask = new_image[:, :, 3].copy()
  28. mask = cv2.floodFill(mask, None, (int(w/2+t), int(h/2+t)), 128)[1]
  29. mask[mask != 128] = 0
  30. mask[mask == 128] = 1
  31. mask = np.stack((mask, mask, mask), axis=2)
  32. # Blend images
  33. temp = np.zeros_like(new_image[:, :, :3])
  34. temp[(t-1):(h+t-1), (t-1):(w+t-1)] = image.copy()
  35. new_image[:, :, :3] = new_image[:, :, :3] * (1 - mask) + temp * mask
  36. # Set proper alpha channel in new image
  37. temp = new_image[:, :, 3].copy()
  38. new_image[:, :, 3] = cv2.floodFill(temp, None, (int(w/2+t), int(h/2+t)), 255)[1]
  39. return new_image
  40. img = cv2.imread('path/to/your/image.png')
  41. cv2.imshow('img', img)
  42. new_img = rect_with_rounded_corners(img, 50, 20, (0, 0, 0))
  43. cv2.imshow('new_img', new_img)
  44. cv2.waitKey(0)
  45. cv2.destroyAllWindows()

字符串
这和我在另一个答案中使用的概念是一样的,只是在正确的透明度问题上多写了一些代码。
一些示例性输入:
x1c 0d1x的数据
相应的输出:



另一个输入和参数集:


  1. new_img = rect_with_rounded_corners(img, 20, 10, (0, 0, 128))


输出量:

希望也有帮助!

  1. ----------------------------------------
  2. System information
  3. ----------------------------------------
  4. Platform: Windows-10-10.0.16299-SP0
  5. Python: 3.8.1
  6. NumPy: 1.18.1
  7. OpenCV: 4.2.0
  8. ----------------------------------------

展开查看全部
qeeaahzv

qeeaahzv2#

我喜欢用SVG画圆角矩形,这是一种改变--不仅仅是因为有人认为我总是使用ImageMagick ;-)

  1. #!/usr/bin/env python3
  2. from PIL import ImageOps, Image
  3. from cairosvg import svg2png
  4. from io import BytesIO
  5. def frame(im, thickness=5):
  6. # Get input image width and height, and calculate output width and height
  7. iw, ih = im.size
  8. ow, oh = iw+2*thickness, ih+2*thickness
  9. # Draw outer black rounded rect into memory as PNG
  10. outer = f'<svg width="{ow}" height="{oh}" style="background-color:none"><rect rx="20" ry="20" width="{ow}" height="{oh}" fill="black"/></svg>'
  11. png = svg2png(bytestring=outer)
  12. outer = Image.open(BytesIO(png))
  13. # Draw inner white rounded rect, offset by thickness into memory as PNG
  14. inner = f'<svg width="{ow}" height="{oh}"><rect x="{thickness}" y="{thickness}" rx="20" ry="20" width="{iw}" height="{ih}" fill="white"/></svg>'
  15. png = svg2png(bytestring=inner)
  16. inner = Image.open(BytesIO(png)).convert('L')
  17. # Expand original canvas with black to match output size
  18. expanded = ImageOps.expand(im, border=thickness, fill=(0,0,0)).convert('RGB')
  19. # Paste expanded image onto outer black border using inner white rectangle as mask
  20. outer.paste(expanded, None, inner)
  21. return outer
  22. # Open image, frame it and save
  23. im = Image.open('monsters.jpg')
  24. result = frame(im, thickness=10)
  25. result.save('result.png')

字符串

输出图片

x1c 0d1x的数据

输入图片



您可以使用rxry来更改角的半径。
这里是outerinnerexpanded-正如你所看到的,它们都是相同的大小,以便在彼此之上进行组合。



其他想法:

  • 你也可以通过在一个黑盒子里画一个白色矩形,然后在它上面运行一个中值滤波器,或者一些形态学侵 eclipse ,来创建一个圆角。如果你过滤这个:

使用15 x15的中值滤波器,你会得到:

如果有人想要一个ImageMagick解决方案:

  1. #!/bin/bash
  2. # Get width and height of input image
  3. read iw ih < <(identify -format "%w %h" monsters.jpg)
  4. # Calculate size of output image, assumes thickness=10
  5. ((ow=iw+20))
  6. ((oh=ih+20))
  7. magick -size ${ow}x${oh} xc:none -fill black -draw "roundrectangle 0,0 $ow,$oh 20,20" \
  8. \( -size ${iw}x${ih} xc:black -fill white -draw "roundrectangle 0,0,$iw,$ih 20,20" monsters.jpg -compose darken -composite \) \
  9. -gravity center -compose over -composite result.png

Keywords:Python,图像处理,圆角,圆角,边框,SVG,cairo,cairosvg,SVG to PNG,SVG as PNG,SVG to PIL,PIL,Pillow。

展开查看全部
hlswsv35

hlswsv353#

当然,Mark会使用ImageMagick提供一个很好的解决方案。但是,因为你的问题是用Pillow标记的,其他人可能也在寻找解决方案,这里是我的手动实现,因为我怀疑,有一个现成的内置方法:

  1. from matplotlib import pyplot as plt # Just for visualization
  2. from PIL import Image, ImageDraw
  3. def rect_with_rounded_corners(image, r, t, c):
  4. """
  5. :param image: PIL image, assumption: uni color filled rectangle
  6. :param r: radius of rounded corners
  7. :param t: thickness of border
  8. :param c: color of border
  9. :return: new PIL image of rectangle with rounded corners
  10. """
  11. # Some method to extract the main color of the rectangle needed here ...
  12. mc = img.getpixel((image.width/2, image.height/2))
  13. # Create new image
  14. new_image = Image.new(image.mode, (image.width + 2*t, image.height + 2*t), (255, 255, 255))
  15. draw = ImageDraw.Draw(new_image)
  16. # Draw four rounded corners
  17. draw.arc([(0, 0), (2*r-1, 2*r-1)], 180, 270, c, t)
  18. draw.arc([(image.width-2*r+2*t, 0), (image.width+2*t, 2*r-1)], 270, 0, c, t)
  19. draw.arc([(image.width-2*r+2*t, image.height-2*r+2*t), (image.width+2*t, image.height+2*t)], 0, 90, c, t)
  20. draw.arc([(0, image.height-2*r+2*t), (2*r-1, image.height+2*t)], 90, 180, c, t)
  21. # Draw four edges
  22. draw.line([(r-1, t/2-1), (image.width-r+2*t, t/2-1)], c, t)
  23. draw.line([(t/2-1, r-1), (t/2-1, image.height-r+2*t)], c, t)
  24. draw.line([(image.width+1.5*t, r-1), (image.width+1.5*t, image.height-r+2*t)], c, t)
  25. draw.line([(r-1, image.height+1.5*t), (image.width-r+2*t, image.height+1.5*t)], c, t)
  26. # Fill rectangle with main color
  27. ImageDraw.floodfill(new_image, (image.width/2+t, image.height/2+t), mc)
  28. return new_image
  29. img = Image.new('RGB', (640, 480), (255, 128, 255))
  30. plt.figure(1)
  31. plt.imshow(img)
  32. new_img = rect_with_rounded_corners(img, 100, 20, (0, 0, 0))
  33. plt.figure(2)
  34. plt.imshow(new_img)
  35. plt.show()

字符串
基本上,它是计算和手动绘制四个弧,四个边缘与所需的厚度和颜色的边界,然后泛色填充矩形与初始矩形的颜色。把它放在一些方法,并在需要时重用它,所以没有混乱的主要代码。
对于指定的图像和参数集,我们得到输出(Matplotlib图在这里):
x1c 0d1x的数据
对于另一个图像和参数集,

  1. img = Image.new('RGB', (400, 300), (0, 64, 255))
  2. plt.figure(1)
  3. plt.imshow(img)
  4. new_img = rect_with_rounded_corners(img, 25, 10, (255, 0, 0))
  5. plt.figure(2)
  6. plt.imshow(new_img)


我们得到,例如:



希望能帮上忙!

  1. ----------------------------------------
  2. System information
  3. ----------------------------------------
  4. Platform: Windows-10-10.0.16299-SP0
  5. Python: 3.8.1
  6. Matplotlib: 3.2.0rc3
  7. Pillow: 7.0.0
  8. ----------------------------------------

展开查看全部
8ljdwjyq

8ljdwjyq4#

这里是另一种使用Python/OpenCV的方法。然而,在这种方法中,边框将在输入图像的边界内。

  • 读取输入
  • 创建输入大小的白色图像
  • 用黑色填充白色图像,使其周围达到所需的边框厚度
  • 对填充图像应用高斯模糊
  • 对模糊图像进行阈值处理,形成二值图像
  • 腐 eclipse 阈值图像以形成第二二个二值图像
  • 得到两个二值图像之间的差以形成边界形状掩模
  • 根据厚度修剪边框蒙版,使其恢复到输入图像的大小
  • 创建与输入大小相同的彩色图像
  • 使用蒙版将输入和彩色图像合并组合
  • 将第一个阈值化图像放入组合图像的alpha通道中,使外部透明
  • 保存结果

输入:
x1c 0d1x的数据

  1. import cv2
  2. import numpy as np
  3. # set thickness, rounding and color of border
  4. t = 21
  5. r = 21
  6. c = (0,0,255)
  7. # read image
  8. img = cv2.imread("bear.png")
  9. hh, ww = img.shape[0:2]
  10. # create white image of size of input
  11. white = np.full_like(img, (255,255,255))
  12. # add black border of thickness
  13. border = cv2.copyMakeBorder(white, t, t, t, t, borderType=cv2.BORDER_CONSTANT, value=(0,0,0))
  14. # blur image by rounding amount as sigma
  15. blur = cv2.GaussianBlur(border, (0,0), r, r)
  16. # threshold blurred image
  17. thresh1 = cv2.threshold(blur, 128, 255, cv2.THRESH_BINARY)[1]
  18. # create thesh2 by eroding thresh1 by 2*t
  19. kernel = cv2.getStructuringElement(cv2.MORPH_ELLIPSE, (2*t,2*t))
  20. thresh2 = cv2.morphologyEx(thresh1, cv2.MORPH_ERODE, kernel, iterations=1)
  21. # subtract the two thresholded images to make a border mask
  22. mask = thresh1 - thresh2
  23. # shave border mask by t
  24. mask = mask[t:hh+t,t:ww+t]
  25. # create colored image the same size as input
  26. color = np.full_like(img, c)
  27. # combine input and color with mask
  28. result = cv2.bitwise_and(color, mask) + cv2.bitwise_and(img, 255-mask)
  29. # add thresh1 as alpha channel
  30. thresh1 = thresh1[t:hh+t,t:ww+t][:,:,0]
  31. result = np.dstack([result,thresh1])
  32. # write
  33. cv2.imwrite("bear_thresh1.png", thresh1)
  34. cv2.imwrite("bear_thresh2.png", thresh2)
  35. cv2.imwrite("bear_mask.png", mask)
  36. cv2.imwrite("bear_red_border.png", result)
  37. # display it
  38. cv2.imshow("IMAGE", img)
  39. cv2.imshow("BORDER", border)
  40. cv2.imshow("BLUR", blur)
  41. cv2.imshow("THRESHOLD1", thresh1)
  42. cv2.imshow("THRESHOLD2", thresh2)
  43. cv2.imshow("MASK", mask)
  44. cv2.imshow("RESULT", result)
  45. cv2.waitKey(0)

字符串
阈值1图像:



阈值2图像:



边框蒙版图像:

结果图像:

添加

这里是对上面的修正,允许更多的厚度和半径值。示例使用厚度21和半径81。

  1. import cv2
  2. import numpy as np
  3. import skimage.exposure
  4. # set thickness, rounding and color of border
  5. t = 21
  6. r = 81
  7. c = (0,0,255)
  8. # read image
  9. img = cv2.imread("bear.png")
  10. hh, ww = img.shape[0:2]
  11. # create white image of size of input
  12. white = np.full_like(img, (255,255,255))
  13. # add black border of thickness r
  14. border = cv2.copyMakeBorder(white, r,r,r,r, borderType=cv2.BORDER_CONSTANT, value=(0,0,0)).astype(np.float64)
  15. # blur image by rounding amount as sigma
  16. blur = cv2.GaussianBlur(border, (0,0), r, r)
  17. # threshold blurred image
  18. thresh1 = cv2.threshold(blur, 128, 255, cv2.THRESH_BINARY)[1]
  19. # create thesh2 by eroding thresh1 by 2*t
  20. kernel = cv2.getStructuringElement(cv2.MORPH_ELLIPSE, (2*t,2*t))
  21. thresh2 = cv2.morphologyEx(thresh1, cv2.MORPH_ERODE, kernel, iterations=1)
  22. # subtract the two thresholded images to make a border mask
  23. mask = thresh1 - thresh2
  24. # antialias
  25. mask = mask.astype(np.float64)
  26. smooth = cv2.GaussianBlur(mask, (0,0), sigmaX=3, sigmaY=3, borderType = cv2.BORDER_DEFAULT)
  27. mask = skimage.exposure.rescale_intensity(smooth, in_range=(96,160), out_range=(0,255))
  28. # shave border mask by r
  29. mask = mask[r:hh+r,r:ww+r]
  30. # create colored image the same size as input
  31. color = np.full_like(img, c)
  32. # combine input and color with mask
  33. #result = cv2.bitwise_and(color, mask) + cv2.bitwise_and(img, 255-mask)
  34. result = ((color*mask + img*(255-mask))/255).clip(0,255).astype(np.uint8)
  35. # add thresh1 as alpha channel
  36. thresh1 = thresh1[r:hh+r,r:ww+r][:,:,0]
  37. result = np.dstack([result,thresh1])
  38. # write
  39. cv2.imwrite("bear2_thresh1.png", thresh1)
  40. cv2.imwrite("bear2_thresh2.png", thresh2)
  41. cv2.imwrite("bear2_mask.png", mask)
  42. cv2.imwrite("bear2_red_border.png", result)
  43. # display it
  44. cv2.imshow("IMAGE", img)
  45. cv2.imshow("BORDER", border)
  46. cv2.imshow("BLUR", blur)
  47. cv2.imshow("THRESHOLD1", thresh1)
  48. cv2.imshow("THRESHOLD2", thresh2)
  49. cv2.imshow("MASK", mask)
  50. cv2.imshow("RESULT", result)
  51. cv2.waitKey(0)


测试结果:

展开查看全部
csga3l58

csga3l585#

这里是一个简单的实现只是枕头:

  1. from PIL import Image, ImageDraw
  2. def add_round_border(
  3. image, border_color=(232, 232, 232), border_radius=30, border_width=3
  4. ):
  5. image = image.convert("RGBA")
  6. # Create an out mask and an in mask
  7. mask = Image.new("L", image.size, 0)
  8. draw = ImageDraw.Draw(mask)
  9. draw.rounded_rectangle(
  10. [0, 0, image.size[0], image.size[1]], radius=border_radius, fill=255
  11. )
  12. mask_in = Image.new("L", image.size, 0)
  13. draw = ImageDraw.Draw(mask_in)
  14. draw.rounded_rectangle(
  15. [
  16. border_width,
  17. border_width,
  18. image.size[0] - border_width,
  19. image.size[1] - border_width,
  20. ],
  21. radius=border_radius - border_width,
  22. fill=255,
  23. )
  24. border_image = Image.new("RGBA", image.size, color=border_color)
  25. new_image = Image.new("RGBA", image.size, color=0)
  26. # Add the border by pasting the border images onto the new image
  27. new_image.paste(border_image, mask=mask)
  28. new_image.paste(image, mask=mask_in)
  29. return new_image
  30. # Driver Code
  31. image = Image.open("YOUR_IMAGE_PATH")
  32. image_with_border = add_round_border(image, border_color="red", border_radius=30, border_width=15)
  33. image.save("test.png")
  34. image_with_border.save("test_with_border.png")

字符串
之前:https://i.stack.imgur.com/Cm92N.jpg
后:https://i.stack.imgur.com/3HIuK.jpg

展开查看全部

相关问题