哈哈哈哈哈哈镜~
时间:2022-07-23
本文章向大家介绍哈哈哈哈哈哈镜~,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
我绝对是无聊爆炸了,所以我又丧心病狂处理二次元图片了。
今天基于像素变换来实现图像的哈哈镜变换,效果就是下面这样了:
哈哈镜分两种,一种是挤压,一种是放大。分别对应凹函数和凸函数。
输入一副图像,首先设置缩放中心center,我们取原图鼻子处为中心。
设置图像上任意一点到中心点的相对坐标tx= x-cx,ty=y-cy。
左边为挤压哈哈镜,对应像素映射函数:
//变换后新的坐标
x = cos(atan2(ty , tx))* 12*(sqrt(tx*tx + ty*ty))+cx
y = sin(atan2(ty , tx))* 12*(sqrt(tx*tx + ty*ty))+cy
公式中的常数12代表强度,越大则图像越扭曲。
自定义挤压函数(C++版)(Python版可以点击文末阅读原文查看):
Mat MinFrame(Mat frame) {
Mat srcImage;
frame.copyTo(srcImage);
int radius = 400;//定义哈哈镜的半径
int height = frame.rows;
int width = frame.cols; //图片的长宽
Point2d center;//人脸中心
center.x = 130;
center.y = 180;
int newX, newY;//变换后的坐标
for (int x = 0; x < width; x++)
for (int y = 0; y < height; y++) {
double tX = x - center.x;
double tY = y - center.y;
double theta = atan2(tY, tX);
radius = sqrt((tX * tX) + (tY * tY)); //与上面一样,计算公式不一样
int newR = sqrt(radius) *8;
newX = int(center.x + (newR * cos(theta)));
newY = int(center.y + (newR * sin(theta)));
if (newX<0 && newX>width)
newX = 0;
if (newY<0 && newY>height)
newY = 0;
if (newX < width && newY < height) {
srcImage.at<Vec3b>(y, x)[0] = frame.at<Vec3b>(newY, newX*3)[0]; //将计算后的坐标移动到原坐标
srcImage.at<Vec3b>(y, x)[1] = frame.at<Vec3b>(newY, newX*3+1)[1];
srcImage.at<Vec3b>(y, x)[2] = frame.at<Vec3b>(newY, newX*3+2)[2];
}
else {
srcImage.at<Vec3b>(y, x)[0] = frame.at<Vec3b>(y, x)[0]; //将计算后的坐标移动到原坐标
srcImage.at<Vec3b>(y, x)[1] = frame.at<Vec3b>(y, x)[1];
srcImage.at<Vec3b>(y, x)[2] = frame.at<Vec3b>(y, x)[2];
}
}
return srcImage;
}
右边为放大哈哈镜,对应像素映射函数:
//变换后的新坐标
x = (tx/2)*(sqrt(tx*tx + ty*ty)/radius)+cx
y = (ty/2)*(sqrt(tx*tx + ty*ty)/radius)+cy
自定义放大函数(C++版):
Mat MaxFrame(Mat frame) {
Mat srcImage;
frame.copyTo(srcImage);
int radius = 400;//定义哈哈镜的半径
int height = frame.rows;
int width = frame.cols; //图片的长宽
Point2d center;//图片中心
center.x = 130;//人脸中心像素坐标
center.y = 180;
int newX, newY;//变换后的坐标
int real_radius = int(radius / 2.0); //计算公式
for (int x = 0; x < width; x++)
for (int y = 0; y < height; y++) {
int tX = x - center.x;
int tY = y - center.y;
int distance = tX * tX + tY * tY;
if (distance < radius*radius) {
newX = int(tX / 2.0);
newY = int(tY / 2.0);
newX = int(newX * (sqrt(distance) / real_radius));
newY = int(newY * (sqrt(distance) / real_radius));
newX = int(newX + center.x);
newY = int(newY + center.y);
if (newX < width && newY < height) {
srcImage.at<Vec3b>(y, x)[0] = frame.at<Vec3b>(newY, newX)[0]; //将计算后的坐标移动到原坐标
srcImage.at<Vec3b>(y, x)[1] = frame.at<Vec3b>(newY, newX)[1];
srcImage.at<Vec3b>(y, x)[2] = frame.at<Vec3b>(newY, newX)[2];
}
}
else {
srcImage.at<Vec3b>(y, x)[0] = frame.at<Vec3b>(y, x)[0]; //将计算后的坐标移动到原坐标
srcImage.at<Vec3b>(y, x)[1] = frame.at<Vec3b>(y, x)[1];
srcImage.at<Vec3b>(y, x)[2] = frame.at<Vec3b>(y, x)[2];
}
}
return srcImage;
}
至于这俩函数怎么来的,,我也说不出啥门道来,,欢迎评论告知哈。
再看最后一眼:
THE END
- 沙特机器人获得公民身份?强人工智能时代还早呢!
- 整理了二个基本的css库(高手请绕道)
- 省钱提效做管理,医疗保健初创公司保持AI优势的四个方法
- Linux操作系统启动流程梳理
- js中数组(Array)的排序(sort)注意事项
- 双机热备工作模式及高内聚低耦合架构解释
- linux下拷贝命令中的文件过滤操作记录
- 关于智慧城市的十大反思(上)
- scrollTop与offsetTop研究
- JQuery笔记(四) 通用选择的尝试
- Docker容器学习梳理--基础环境安装
- Javascript:模仿淘宝的信用评价
- 好米有好价! 两枚4字母域名均以五位数交易
- Docker容器学习梳理--Volume数据卷使用
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- android实现倒计时功能(开始、暂停、0秒结束)
- 【-Flutter/Dart 语法补遗-】 sync* 和 async* 、yield 和yield* 、async 和 await
- android实现条目倒计时功能
- Android实现简单手电筒功能
- k8s安装自动证书签发cert-manager letsencrypt
- Android仿Keep运动休息倒计时圆形控件
- android通过led实现手电筒功能
- Android 7.0 手电筒控制实现
- 【STM32H7】第13章 RL-TCPnet V7.X之创建多个TCP客户端
- Android倒计时的开始与停止 剩余时分秒的展示
- 由LFI引起的Zimbra邮件管理系统0day
- Android手电筒兼容各个手机与版本
- 【STM32F429】第13章 RL-TCPnet V7.X之创建多个TCP客户端
- RecyclerView仿应用列表实现网格布局
- Android实现带进度条的WebView