实现抖音哈哈镜效果---OpenCV-Python开发指南(47)

举报
择城终老 发表于 2021/07/26 23:14:50 2021/07/26
【摘要】 目录 前言哈哈镜放大效果哈哈镜缩小效果直接视频实现哈哈镜效果 前言 在抖音中,我们经常看到各种抖音玩家都喜欢使用哈哈镜效果。那么什么是哈哈镜呢? 在现实生活中,哈哈镜是指一种表面凹凸不平的镜子,可以反应出人像及物体的扭曲面貌。简单点来说就是拉伸人脸(物件),或者压缩人脸(物体)的效果。 哈哈镜的实现原理如下: 假设输入图像的宽高为w和h,图像中心点...

前言

在抖音中,我们经常看到各种抖音玩家都喜欢使用哈哈镜效果。那么什么是哈哈镜呢?

在现实生活中,哈哈镜是指一种表面凹凸不平的镜子,可以反应出人像及物体的扭曲面貌。简单点来说就是拉伸人脸(物件),或者压缩人脸(物体)的效果。

哈哈镜的实现原理如下:

假设输入图像的宽高为w和h,图像中心点的坐标为(cx,cy)。那么图像中任意一点(x,y)到中心点的距离为(x-cx),(y-cy)。

那么拉升放大,图像变换的半径为r(r是哈哈镜的范围大小),得到公式如下:

x=(tx/2)(sqrt(txtx+tyty)/r)+cx
y=(ty/2)
(sqrt(txtx+tyty)/r)+cy

同理,压缩缩小的公式如下(compress为压缩强度):
x=cos(atan2(ty,tx))compresssqrt(sqrt(txtx+tyty))+cx
y=cos(atan2(ty,tx))compresssqrt(sqrt(txtx+tyty))+cy

哈哈镜放大效果

既然我们了解了其数学公式以及其实现的原理,下面我们来直接实现哈哈镜的放大效果。具体代码如下所示:

#哈哈镜放大效果实现
def enlarge_effect(img): h, w, n = img.shape cx = w / 2 cy = h / 2 radius = 100#该值可以自行定义,它决定了哈哈镜的大小,当图像很大时,应该相应的调大 r = int(radius / 2.0) new_img = img.copy() for i in range(w): for j in range(h): tx = i - cx ty = j - cy distance = tx * tx + ty * ty if distance < radius * radius: x = int(int(tx / 2.0) * (math.sqrt(distance) / r) + cx) y = int(int(ty / 2.0) * (math.sqrt(distance) / r) + cy) if x < w and y < h: new_img[j, i, 0] = img[y, x, 0] new_img[j, i, 1] = img[y, x, 1] new_img[j, i, 2] = img[y, x, 2] return new_img


if __name__ == "__main__": img = cv2.imread("4.jpg") enlarge_img = enlarge_effect(img) cv2.imshow("4", enlarge_img) cv2.waitKey() cv2.destroyAllWindows()

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29

运行之后,效果如下:
哈哈镜放大效果

需要注意的是,上面的计算过程可能导致有浮点数的出现,而像素值必须为整数。所以,为了保证像素值的有效性,必须在计算过程完整之后,进行强制类型转换int()。另外,计算x,y值时,可能会导致超过图像坐标的范围,所以必须用x<w和y<h来判断防止越界。

哈哈镜缩小效果

下面我们直接实现哈哈镜的缩小效果。具体代码如下所示:

def reduce_effect(img): h, w, n = img.shape cx = w / 2 cy = h / 2 radius = 100 r = int(radius / 2.0) compress = 8 new_img = img.copy() for i in range(w): for j in range(h): tx = i - cx ty = j - cy x = int(cx + (math.sqrt(math.sqrt(tx * tx + ty * ty)) * compress * math.cos(math.atan2(ty, tx)))) y = int(cy + (math.sqrt(math.sqrt(tx * tx + ty * ty)) * compress * math.sin(math.atan2(ty, tx)))) if x < 0 and x > w: x = 0 if y < 0 and y > h: y = 0 if x < w and y < h: new_img[j, i, 0] = img[y, x, 0] new_img[j, i, 1] = img[y, x, 1] new_img[j, i, 2] = img[y, x, 2] return new_img

if __name__ == "__main__": img = cv2.imread("4.jpg") enlarge_img = enlarge_effect(img) frame = reduce_effect(img) cv2.imshow("1", img) cv2.imshow("2", enlarge_img) cv2.imshow("3", frame) cv2.waitKey() cv2.destroyAllWindows()

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33

运行之后,效果如下:
效果大小

直接视频实现哈哈镜效果

抖音上面的哈哈镜都是动态的,并不是单一的图片这么变来变去。其实,只要我们集合前面的摄像头视频录制功能,就可以完成视频哈哈镜的动态效果。具体代码如下:

if __name__ == "__main__":
	cap = cv2.VideoCapture(0) while (cap.isOpened()): ret, frame = cap.read() frame=enlarge_effect(frame) cv2.imshow('video', frame) c = cv2.waitKey(1) if c == 27: break cap.release() cv2.destroyAllWindows()

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11

文章来源: liyuanjinglyj.blog.csdn.net,作者:李元静,版权归原作者所有,如需转载,请联系作者。

原文链接:liyuanjinglyj.blog.csdn.net/article/details/115091122

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。