1前言

    我们在工作和生活中经常要填写一些个人资料,这时候往往需要放证件照上去,但是有时候人家要求是红底或白底,但是偏偏不巧的是你以前照了张蓝底的。这时候你想换个底色,于是在百度上一搜“证件照换底色”,出来了一堆photoshop 教程,程序员可以找公司美工帮个忙,但是一般人就很纠结了,所以呢我就花了半天的功夫写了一个小软件,简化大家的操作难度,哎!现在越来越发现会写点程序真好。O(∩_∩)O哈哈~

2.实现思路

(1)首先说一下像素的组成RGB

当前展示白色的RGB值是255,255,255,相反黑色的RGB就是0,0,0,  其他颜色就是0~255的RGB颜色组合。我们就是通过颜色的RGB值得范围来替换证件照底色的。

(2要过滤底色的RGB范围选择)

我们一般拍照的时候后面都有一块幕布做底色,但是由于曝光和幕布底色的稍微的不同,导致背景其实是一个颜色范围,而非单一的颜色,这就要求我们必须找出这个范围,我这里选择前五排的像素作为初始过滤的颜色范围。下面就是代码获取这个颜色范围。

        /// <summary>
/// 获取前五排像素,求出rgb范围
/// </summary>
/// <param name="path"></param>
/// <returns></returns>
private List<int> GetModeColor(string path)
{
List<int> colorrgb = new List<int>();
List<int> rlist=new List<int>();
List<int> glist = new List<int>();
List<int> blist = new List<int>();
using (Bitmap map = (Bitmap)Image.FromFile(path))
{
using (Bitmap editMap = new Bitmap(map, map.Width, map.Height))
{
// editMap.GetPixel()
for (int i = ; i < map.Width; i++)
{
for (int j = ; j <; j++)
{
Color color = editMap.GetPixel(i, j);
if (!rlist.Contains(color.R))
{
rlist.Add(color.R);
}
if (!glist.Contains(color.G))
{
glist.Add(color.G);
}
if (!blist.Contains(color.B))
{
blist.Add(color.B);
}
}
}
//添加rgb像素范围
colorrgb.Add(rlist.Max());
colorrgb.Add(glist.Max());
colorrgb.Add(blist.Max());
colorrgb.Add(rlist.Min());
colorrgb.Add(glist.Min());
colorrgb.Add(blist.Min());
}
}
return colorrgb;
}

(3头部和身体进行分开处理)

由于我们的头发偏黑色,若要达到很好的处理效果必须进行对头部做更大范围的颜色处理,这里我们加入了一个选择条,让用户进行分区域进行微调,在文本框中输入要调整颜色范围数值,达到最好的效果。

(4背景颜色移除与替换)

        我们通过对证件照从左到右,从上到下一行一行的对颜色进行过滤,把与我们设置颜色范围内的像素移除并替换成我们的想要的背景色,经过微调,最大程度的实现颜色的替换而不失去本该保留部分。这也是本工具的最核心代码。

 /// <summary>
/// 预览结果
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private void button3_Click(object sender, EventArgs e)
{
if (string.IsNullOrEmpty(label1.Text))
{
MessageBox.Show("请选择颜色");
return;
} if (string.IsNullOrEmpty(imagePath))
{
MessageBox.Show("请上传证件");
return;
} int upnum = Convert.ToInt32(textBox1.Text);
int downnum = Convert.ToInt32(textBox2.Text);
double trackvalue = Convert.ToDouble(trackBar1.Value);
List<int> colorrgb = GetModeColor(imagePath);
//上半部分去除颜色范围
List<int> upcolorrgb = new List<int>();
//下半部分去除颜色范围
List<int> downcolorrgb = new List<int>();
for (int i = ; i < colorrgb.Count; i++)
{
if (i>)
{ upcolorrgb.Add(colorrgb[i]-upnum<?: colorrgb[i] - upnum);
downcolorrgb.Add(colorrgb[i] - downnum < ? : colorrgb[i] - downnum);
}
else
{
upcolorrgb.Add(colorrgb[i] + upnum > ? : colorrgb[i] + upnum);
downcolorrgb.Add(colorrgb[i] + downnum > ? : colorrgb[i] + downnum);
}
}
//从左到右,从上到下读取像素点
using (Bitmap map = (Bitmap)Image.FromFile(imagePath))
{
using (Bitmap editMap = new Bitmap(map, map.Width, map.Height))
{
// 上半部分
for (int i = ; i < map.Width; i++)
{
for (int j = ; j < (int)(map.Height * (trackvalue/)); j++)
{
Color color = editMap.GetPixel(i, j);
//判断像素是否可以移除
if (color.R >= upcolorrgb[] && color.R <= upcolorrgb[] &&
color.G >= upcolorrgb[] && color.G <= upcolorrgb[] &&
color.B >= upcolorrgb[] && color.B <= upcolorrgb[])
{
editMap.SetPixel(i, j, tempcolor); }
}
}
//下半部分
for (int i = ; i < map.Width; i++)
{
for (int j = (int)(map.Height * (trackvalue / )); j <map.Height ; j++)
{
//判断像素是否可以移除
Color color = editMap.GetPixel(i, j);
if (color.R >= downcolorrgb[] && color.R <= downcolorrgb[] &&
color.G >= downcolorrgb[] && color.G <= downcolorrgb[] &&
color.B >= downcolorrgb[] && color.B <= downcolorrgb[])
{
editMap.SetPixel(i, j, tempcolor); }
}
}
//保存
string savepath = System.Environment.CurrentDirectory+@"\result\" + DateTime.Now.ToString("yyyyMMddHHmmssffff") + ".png";
editMap.Save(savepath);
resultPath = savepath;
pictureBox2.Image = Image.FromFile(savepath);
}
}
}

3.软件操作

(0系统界面)

(1打开证件照)

由于本人太丑,这里在网上找了张美女的证件照,来做演示。

(2选择替换后的背景色)

点击-打开选择颜色,选择想要的背景色,

(3预览初始效果)

点击预览,便可看到最初的处理效果。

(4微调)

调节滑块,将图像分为上下两部分分开进行处理,调节参数(范围0-255都行),参数值越大移除的细节越多,仔细调参数可以把头部保留更多细节。

换个红色看看效果

(5导出)

点击导出成果就可以啦。这里就不放图了。完毕

4.总结

       可能存在的一点小问题就是衣服的颜色和背景相似,可能会被误移除,一般的情况都会有比较满意的效果。这是我2018年的第一篇小文章,一步步实现自己的小目标。希望大家多多支持。下面是软件和源码的下载。

源码:https://pan.baidu.com/s/1kVBvUD5

工具程序:https://pan.baidu.com/s/1nvxKFeL

作者:ATtuing

出处:http://www.cnblogs.com/ATtuing

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文链接。

c#实现超实用的<证件照换底色>小工具的更多相关文章

  1. 教你用OpenCV 和 Python给证件照换底色(蓝底 <->红底->白底)

    在我们的生活中常常要用到各种底色要求的证件电子照,红底.蓝底.或者白底,而假如你手上只有一种底色的证件照,你又不想再去拍又不会PS怎么办?今天教你们用OpenCV和Python给你的证件照换底色. P ...

  2. 20行代码教你用python给证件照换底色

    1.图片来源 该图片来源于百度图片,如果侵权,请联系我删除!图片仅用于知识交流.本文只是为了告诉大家:python其实有很多黑科技(牛逼的库),我们既可以用python处理工作中的一些事儿,同时我们也 ...

  3. opencv —— floodFill 漫水填充法 实现证件照换背景

    漫水填充:floodFill 函数 简单来说,漫水填充就是自动选中与种子像素相连的区域,利用指定颜色进行区域颜色填充.Windows 画图工具中的油漆桶功能和 Photoshop 的魔法棒选择工具,都 ...

  4. 【原】得心应手小工具开发——IE代理快速切换工具

    一.引入 因为公司里上外网要经常换IE代理地址,每次切换地址都要进到Internet Options里去设置一番,经常切换的话很是麻烦,由于用了点时间作个小工具来方便自己. 二.实现思路 其实思路很简 ...

  5. ContentProvider域名替换小工具

    开发项目域名想怎么换就怎么换,就是这么任性! 这是一个很有意思的小工具! 这是一个方便开发人员和测试人员的小工具!! 吐槽: 一直在做Android开发,一直总有一个问题存在:做自己公司的apk开发时 ...

  6. Windows下好用到必须开机自启的小工具

    折腾过linux,黑苹果,最后还是回到了盖茨大叔的windows.得出的结论是,日常使用的话,折腾Linux还不如把精力去拿去折腾windows.分享下折腾的成果,介绍下一些很不错的小工具.     ...

  7. bug检测报告---礼物挑选小工具--飞天小女警

    飞天小女警----礼物挑选小工具 测试产品链接:http://123.207.159.79:8088/giving_gifts/ 发布在作者的博客里面:http://www.cnblogs.com/s ...

  8. 2014年Windows平台软件推荐:神器小工具(骨灰级

    原文  http://www.wtoutiao.com/a/120621.html 底层工具 “If you know how to use Process Monitor competently, ...

  9. python小工具:用python操作HP的Quality Center

    背景是这样的:这个组的测试人员每跑一个case都要上传测试结果附件到QC.每个待测功能模块可能包含几十上百的case.于是手工上传测试结果变成了繁重的体力劳动.令人惊讶的是我们的工具开发组竟然说做不了 ...

随机推荐

  1. RxAndroid中observable的基本使用和表单校验操作

    RxAndroid 响应式编程 类似于监听-观察者模式 在观察者模式中,你的对象需要实现 RxJava 中的两个关键接口:Observable 和 Observer.当 Observable 的状态改 ...

  2. tolua++实现lua层调用c++技术分析

    tolua++技术分析 cocos2dx+lua 前言 一直都使用 cocos2dx + lua 进行游戏开发,用 Lua 开发可以专注于游戏逻辑的实现,另外一方面可以实现热更新:而且 lua 是一个 ...

  3. 教你如何用 lib-flexible 实现移动端H5页面适配

    前话 好久没写教程了(可能会误导新手的菜鸟教程( ̄▽ ̄)"). 这是我的github,欢迎前端大大们和我一起学习交流 https://github.com/pwcong 最近入职公司做前端实 ...

  4. cocos2dx中关于Action动作的相关API的具体介绍

     //CCMoveBy  创建一个移动的动作 //參数1:移动到目标坐标所需的时间 //參数2:目标坐标 //支持reverse 能够获取其反向动作 //CCMoveTo  一样的 //CCAct ...

  5. Android4.0-4.4 加入支持状态栏显示耳机图标方法(支持带不带MIC的两种耳机自己主动识别)

    效果如图: 一. 在frameworks/base/packages/SystemUI/res/values/strings.xml 里加入 <string name="headset ...

  6. Parcel.js + Vue 2.x 极速零配置打包体验

    继 Browserify.Webpack 之后,又一款打包工具 Parcel 横空出世 Parcel.js 的官网有这样的自我介绍 “极速零配置Web应用打包工具” 简单接触了一下,单从效率上来说,确 ...

  7. axios遇到的坑

    axios 可以使用一个config.js 配置文件来管理它的请求信息.具体配置不细说,使用如下.   一,使用配置方式 GET方法: let promise = axios.get(url, con ...

  8. TP3.2.3 接入银联支付

    TP3.2.3 接入银联支付 项目接入银联支付的过程, 在此记录下,希望能帮助开发盆友平坑. 银联SKD链接:https://open.unionpay.com/ajweb/product/newPr ...

  9. 使用javaMail和velocity来发送模板邮件

    之前在ssh项目中有用过javaMail和velocity来发送邮件,实现的效果如下所示. 这类邮件主要用于公司的推广宣传,比如商城的促销等场景. 今天打算将邮件模块也集成到ssm项目,也算是对之前做 ...

  10. gunicorn geventworker 解析

    在前面的文章曾介绍过gunicorn的syncworker,本文介绍其中一种asyncworker:GeventWorker.类图如下:   可见GeventWorker重载了init_process ...