美文网首页python
使用Python,自动为图片戴口罩

使用Python,自动为图片戴口罩

作者: ___n | 来源:发表于2020-02-23 12:25 被阅读0次

借助 face_recognition 库可以轻松的识别出人像,然后根据脸颊的宽度和鼻梁位置计算出口罩的大小、方向和位置,并最终生成出戴上口罩的图片。
整个过程并不复杂,但在坐标计算上要格外小心!

思路

要想实现上面的效果,我们应该怎么做?不妨这么想:

  • 首先是识别出人的鼻子(nose_bridge)和脸轮廓(chin)。

  • 通过脸轮廓确定出脸左点(chin_left_point)、脸底点(chin_bottom_point)和脸右点(chin_right_point)。

  • 由鼻子和脸底点确定口罩大小的高度、中心线。

  • 将口罩左右平均分为两个部分,调整左口罩大小,宽度为脸左点到中心线的距离;调整右口罩大小,宽度为脸右点到中心线的距离;合并左右口罩为新口罩。

  • 旋转新口罩,角度为中心线相对于 y 轴的旋转角。

  • 将新口罩放在原图适当位置。

关于人脸识别,可以使用 face_recognition[2] 库进行识别。关于图像处理,可以使用 Pillow[3] 库进行处理。

代码

有了思路之后,实现就是件相对轻松的事情。不过对库的熟悉和图片的变换计算可能要花些时间。

人脸识别:

import face_recognition

face_image_np = face_recognition.load_image_file('/path/to/face/picture')
face_landmarks = face_recognition.face_landmarks(face_image_np)

借助 face_recognition 库可以轻松的识别出人像,最终得到的 face_landmarks 是一个列表,里面的每个 face_landmark 都表示一个人像数据。

face_landmark 是一个字典,其中的键表示人像特征,值表示该特征的点的列表。比如:

  • 键 nose_bridge 表示鼻梁

  • 键 chin 表示脸颊

我们需要根据每个 face_landmark,给对应的头像戴上口罩。

获得鼻子和脸颊的特征点:

import numpy as np

nose_bridge = face_landmark['nose_bridge']
nose_point = nose_bridge[len(nose_bridge) * 1 // 4]
nose_v = np.array(nose_point)

chin = face_landmark['chin']
chin_len = len(chin)
chin_bottom_point = chin[chin_len // 2]
chin_bottom_v = np.array(chin_bottom_point)
chin_left_point = chin[chin_len // 8]
chin_right_point = chin[chin_len * 7 // 8]

通过上述代码,我们获得了:

  • 表示上鼻梁的一个点:nose_point

  • 表示脸左点:chin_left_point

  • 表示脸右点:chin_right_point

  • 表示脸底点:chin_bottom_point

拆分、缩放和合并口罩:

from PIL import Image

_face_img = Image.fromarray(face_image_np)
_mask_img = Image.open('/path/to/mask/picture')

# split mask and resize
width = _mask_img.width
height = _mask_img.height
width_ratio = 1.2
new_height = int(np.linalg.norm(nose_v - chin_bottom_v))

# left
mask_left_img = _mask_img.crop((0, 0, width // 2, height))
mask_left_width = get_distance_from_point_to_line(chin_left_point, nose_point, chin_bottom_point)
mask_left_width = int(mask_left_width * width_ratio)
mask_left_img = mask_left_img.resize((mask_left_width, new_height))

# right
mask_right_img = _mask_img.crop((width // 2, 0, width, height))
mask_right_width = get_distance_from_point_to_line(chin_right_point, nose_point, chin_bottom_point)
mask_right_width = int(mask_right_width * width_ratio)
mask_right_img = mask_right_img.resize((mask_right_width, new_height))

# merge mask
size = (mask_left_img.width + mask_right_img.width, new_height)
mask_img = Image.new('RGBA', size)
mask_img.paste(mask_left_img, (0, 0), mask_left_img)
mask_img.paste(mask_right_img, (mask_left_img.width, 0), mask_right_img)

上述代码主要做了如下内容:

  • 将口罩左右平均分为两个部分。

  • 调整左口罩大小,宽度为脸左点到中心线的距离*宽度系数 1.2。

  • 调整右口罩大小,宽度为脸右点到中心线的距离*宽度系数 1.2。

  • 合并左右口罩为新口罩。

get_distance_from_point_to_line 用来获取一个点到一条线的距离,具体实现可看源代码。

width_ratio 是宽度系数,用来适当扩大口罩。原因我们是根据脸颊的宽度计算口罩的宽度,但口罩是待在耳朵上的,真实宽度应该要更宽。

旋转口罩、并放到原图适当位置:

# rotate mask
angle = np.arctan2(chin_bottom_point[1] - nose_point[1], chin_bottom_point[0] - nose_point[0])
rotated_mask_img = mask_img.rotate(angle, expand=True)

# calculate mask location
center_x = (nose_point[0] + chin_bottom_point[0]) // 2
center_y = (nose_point[1] + chin_bottom_point[1]) // 2

offset = mask_img.width // 2 - mask_left_img.width
radian = angle * np.pi / 180
box_x = center_x + int(offset * np.cos(radian)) - rotated_mask_img.width // 2
box_y = center_y + int(offset * np.sin(radian)) - rotated_mask_img.height // 2

# add mask
_face_img.paste(mask_img, (box_x, box_y), mask_img)

上述代码主要做了如下内容:

  • 旋转新口罩,角度为中心线相对于 y 轴的旋转角。

  • 计算口罩应该放置的坐标。

  • 将新口罩放在原图的计算出的坐标下。

最后就是将新图片保存到本地路径。

以上为核心代码解析,详细的代码请参考如下链接 https://github.com/Prodesire/face-mask


face-mask的使用

①安装 face-mask

pip install face-mask

确保 Python 版本在 3.6 及以上。

②使用 face-mask

直接指定图片路径即可为图片中的人像戴上口罩,并会生成一个新的图片(额外有 -with-mask 后缀):

face-mask /path/to/face/picture

通过指定 --show 选项,还可以使用默认图片查看器打开新生成的图片:

face-mask /path/to/face/picture --show

③效果

给一个人戴上口罩 给多个人戴上口罩 给动漫人物戴上口罩

相关文章

  • 使用Python,自动为图片戴口罩

    借助 face_recognition 库可以轻松的识别出人像,然后根据脸颊的宽度和鼻梁位置计算出口罩的大小、方向...

  • 使用python + selenium实现浏览器自动登录

    使用python + selenium实现浏览器自动登录python版本为3.6.2系统为windows10记录自...

  • 天凉好个秋

    今天又戴口罩了,检查又严格了。其实,天凉戴口罩保暖是很好的。从前我们戴纱布口罩,为挡住北风;冬天近,北风吹,寒...

  • 如何有效防护

    近年,经常会出现雾霾天气。 大家都习惯了雾霾天戴口罩。 可是,像图片中这位美女这样戴口罩的人不在少数。口罩只罩住了...

  • 2020-03-15

    为防止灾疫蔓延,国人响应号召戴上了口罩。 一段时间里,在这个世界上戴口罩的人担心不戴口罩的人没有防范意识,不戴口罩...

  • python 实现图片自动上传七牛返回地址

    python 实现图片自动上传七牛返回地址 使用markdown编写文件,图片插入一直不太方便,有些markdow...

  • 口罩头像一键生成器 || 一键P图口罩头像,口罩一戴,福气常在

    #口罩头像# #一键P图戴口罩# #口罩头像制作# 呐,不允许现在还有不戴口罩的人。 不戴口罩,要么戴呼吸机要么戴...

  • 何时才能摘掉口罩

    自疫情温疫以来 口罩成为必需品 电视上人戴口罩 生活中人戴口罩 都是为不被感染 真想不到口罩的 地位不经意就会 提...

  • 戴口罩耳朵疼

    戴口罩引起的耳朵痛可以通过按摩或者热敷,同时减少使用口罩时间。 1.按摩:戴口罩时间过长可能会导致耳朵的组织受到一...

  • 口罩和头盔

    上半年口罩 下半年头盔 今年是多灾多难保命年 戴口罩为防新冠 戴头盔交通安全 今年口罩摘不掉 骑电动自行车必须戴头...

网友评论

    本文标题:使用Python,自动为图片戴口罩

    本文链接:https://www.haomeiwen.com/subject/jvzdqhtx.html