AI模特换装的前端实现

本文作者为 360 奇舞团前端开发工程师

随着AI的火热发展,涌现了一些AI模特换装的前端工具(比如weshop网站),他们是怎么实现的呢?使用了什么技术呢?下文我们就来探索一下其实现原理。

总体的实现流程如下:我们将下图中的这个模特的图片,使用Segment Anything Model在后端分割图层,然后将分割后的图层mask信息返回给前端处理。在前端中选择需要保留的图层信息(如下图中的模特的衣服图层),然后将选中的图层信息交给后端中的Stable Diffusion处理。后端使用原始图片结合选中的图层蒙版图片结合图生图的功能,可以实现weshop等网站的模特换衣等功能。

dbaf4d8dd8694d2e1908ee85b4e3f674.jpeg

本文先简单介绍一下使用SAM智能图层分割,然后主要介绍一下在前端中怎么对分割后的图层进行选择的处理流程。

使用SAM识别图层

首先我们需要对图层进行分割,在SAM出来之前,我们需要使用PS将模特的衣服选取出来,然后倒出衣服的模板,然后再使用其他工具进行替换。但是现在有了SAM后,我们可以对图片中的事物进去只能区分,获取各种物品的图层。

Segment Anything Model(SAM)是一种尖端的图像分割模型,可以进行快速分割,为图像分析任务提供无与伦比的多功能性。SAM 的先进设计使其能够在无需先验知识的情况下适应新的图像分布和任务,这一功能称为零样本传输。SAM 使任何人都可以在不依赖标记数据的情况下为其数据创建分段掩码。

要深入了解 Segment Anything 模型和 SA-1B 数据集,请访问Segment Anything 网站(https://segment-anything.com/)并查看研究论文Segment Anything(https://arxiv.org/abs/2304.02643)。

我们使用SAM进行图像分割,将一个图片中的物体分割成不同的部分。

def mask2rle(img):'''img: numpy array, 1 - mask, 0 - backgroundReturns run length as string formated'''pixels = img.T.flatten()pixels = np.concatenate([[0], pixels, [0]])runs = np.where(pixels[1:] != pixels[:-1])[0] + 1runs[1::2] -= runs[::2]return ' '.join(str(x) for x in runs)def trans_anns(anns):if len(anns) == 0:returnsorted_anns = sorted(anns, key=(lambda x: x['area']), reverse=False)list = []index = 0# 对每个注释进行处理for ann in sorted_anns:bool_array = ann['segmentation']# 将boolean类型的数组转换为int类型int_array = bool_array.astype(int)# 转化为RLE格式rle = mask2rle(int_array)list.append({"index": index, "mask": rle})index += 1return listimage = cv2.imread('<your image path>')import sys
sys.path.append('<your segment-anything link path>')
from segment_anything import sam_model_registry, SamAutomaticMaskGenerator, SamPredictor# sam 模型路径
sam_checkpoint = '<your sam model path>'
# 根据下载的模型,设置对应的类型
model_type = "vit_h"# device = "cuda"
sam = sam_model_registry[model_type](checkpoint=sam_checkpoint)
# sam.to(device=device)
mask_generator = SamAutomaticMaskGenerator(sam)
masks = mask_generator.generate(image)
# 处理sam返回的图层信息
mask_list = trans_anns(masks)mask_obj = {"height": image.shape[0],"width": image.shape[1],"mask_list": mask_list
}import json
print(json.dumps(mask_obj))

运行以上python代码之前,需要配置sam的python环境,具体的配置描述请查看sam的官方描述。

我们通过以上代码,将我们提供的图片,通过SAM处理后,返回图层分割数据。在trans_anns方法中,将图层按照area从小到大的顺序排序。遍历各个图层,将boolean类型的数组转换为 0 1 int类型,然后对二维numpy array类型的0 1二进制mask图像转换为RLE格式。

RLE是一种简单的无损数据压缩算法,通常用于表示连续的相同值的序列。RLE编码的字符串通常用于在图像分割等任务中存储和传输二进制掩码信息,以便更有效地表示图像中的目标区域。并且方便数据压缩和传输。我们参照的这种编解码方式。也可以使用coco RLE的编解码方式。

将编码后的各图层信息存储到list中,就可以通过接口传输给前端处理了。

前端选择图层

下面这些是本文的重点,在前端将刚才解析后的mask_list信息展示,并可以通过交互选取需要保留的模版,并生成最终合并选取的mask生成一个需要保留的服装模版。

body中的基本组件为

<div id="layer-box" style=" width: 500px; height: 500px;position: relative"><img style="width: 100%; height: 100%; position: absolute" src="https://p0.ssl.qhimg.com/t01989f0d446bed3e58.jpg" /></div><div id="save" @click="save" style="margin-top: 20px;margin-right: 20px; margin-left: 20px;">保存</div><canvas id="mergedCanvas" style="border:1px solid #000;"></canvas>

id为layer-box的div组件作为各个mask的父组件,用于查找和管理各个mask的隐藏和展示。其子组件中的第一个标签是展示原始的模特图片的。

id为save的组件在点击时可以处理保存选中的各个mask为一个新的mask图片,用于处理图片合成。

id为mergedCanvas的canvas是进行图片合成和展示合成后的图片的。

解析SAM处理后的mask_list信息
/*** rle格式图片信息转换为mask信息*/function rle2mask(mask_rle, shape = [500, 500]) {/*mask_rle: run-length as string formatted (start length)shape: [width, height] of array to returnReturns an array, 1 - mask, 0 - background*/const s = mask_rle.split(" ");let starts = s.filter((_, index) => index % 2 === 0).map(Number);const lengths = s.filter((_, index) => index % 2 !== 0).map(Number);starts = starts.map(start => start - 1);const ends = starts.map((start, index) => start + lengths[index]);const img = new Array(shape[0] * shape[1]).fill(0);for (let i = 0; i < starts.length; i++) {for (let j = starts[i]; j < ends[i]; j++) {img[j] = 1;}}// return transposeArray(img, shape);const transposed = new Array(shape[1]).fill(0).map(() => new Array(shape[0]).fill(0));for (let i = 0; i < shape[0]; i++) {for (let j = 0; j < shape[1]; j++) {transposed[j][i] = img[i * shape[1] + j];}}return transposed;}/*** 转换mask图片信息,并设置mask的填充颜色*/function transformMaskImage(item, _width, _height) {let canvas = document.createElement("canvas");let canvasContext = canvas.getContext("2d");canvas.width = _width;canvas.height = _height;let rgbaData = rle2mask(item.mask || '', [_width, _height])for (let y = 0; y < rgbaData.length; y++) {let row = rgbaData[y];for (let x = 0; x < row.length; x++) {let dot = rgbaData[y][x];if (1 === dot && canvasContext) {// 值为1的点填充颜色(canvasContext.fillStyle = "#4169eb"), canvasContext.fillRect(x, y, 1, 1);}}}// canvas当前层的图片(base64格式)// matrix:上边生成的二维数组return { imageData: canvas.toDataURL("image/png"), matrix: rgbaData };}// 使用sam处理后的图层信息(rle编码后的,由于篇幅限制,已省略)const res = { "height": 500, "width": 500, "mask_list": [{ "index": 0, "mask": "109864 3 110361 7 110860 9 111359 10 111859 10 112359 10 112860 9 113360 10 113860 10 114360 10 114860 10 115360 10 115861 8" }, { "index": 1, "mask": "121910 2 122409 4 122908 6 123408 7 123907 8 124407 9 124907 9 125406 11 125905 12 126404 13 126905 12 127405 12 127906 12 128406 12 128907 11 129407 10 129908 8 130408 4" },......] }layers = res.mask_list.map((item) =>transformMaskImage(item, res.width, res.height));

res是sam处理后返回的图层信息(由于篇幅限制,已省略,详情请看demo(https://github.com/yuhao1128/AI-model-mask-select-demo/blob/main/index.html)中的数据)。遍历mask_list,使用canvas保存各个mask的信息。由于前面sam处理后的mask_list是经过压缩编码的,所以在rle2mask方法中对rle编码后的数据解码为 0/1二维数组的格式。rle2mask中的解码方式请参考这种解码(https://www.kaggle.com/code/pestipeti/decoding-rle-masks)方式。

然后遍历二维数组,将值为1的点填充颜色,此处是填充的rgba为"#4169eb"的颜色,可以根据需要自己修改为其他的颜色。此处填充的颜色会在下文中鼠标移动到mask上面时,在mask展示的时候呈现此颜色。

最后在layers中存储各个mask的base64格式的图片信息和二维数组信息。

将各个mask添加到图层
const box = document.querySelector("#layer-box");const baseStyle = "width:100%;height:100%;position: absolute;";//将各个mask添加为layer-box的子组件,并隐藏mask的展示layers.forEach((ele) => {const image = document.createElement("img");image.src = ele.imageData;image.style = `${baseStyle}opacity:0`;image.className = "layer";box.append(image);});

将各个mask添加的图片添加为layer-box组件的子组件,并且设置opacity为0,先隐藏这些mask的展示,在下文会监听鼠标的位置,通过设置mask的opacity属性来展示mask。

监听鼠标的位置和点击
// 鼠标移入mask组件的区域时,展示maskbox.addEventListener("mousemove", (e) => {const { clientX, clientY } = e;const X = box.getBoundingClientRect().left + document.body.scrollLeft;const Y = box.getBoundingClientRect().top + document.body.scrollTop;const x = parseInt(res.width * (clientX - X) / box.getBoundingClientRect().width)const y = parseInt(res.height * (clientY - Y) / box.getBoundingClientRect().height)const allLayers = box.querySelectorAll(".layer");const index = layers.findIndex((item) => item.matrix?.[y]?.[x]);allLayers.forEach((ele, i) => {if (i === index) {ele.style = `${baseStyle}opacity:0.7`;} else {// 已经选中的不需要隐藏if (selectedIndexList.indexOf(i) === -1) {ele.style = `${baseStyle}opacity:0`;}}});});// 鼠标移出mask组件的区域时,隐藏maskbox.addEventListener("mouseout", (e) => {console.log('mouseout selectedIndexList', selectedIndexList);const allLayers = box.querySelectorAll(".layer");allLayers.forEach((ele, i) => {// 只有选中的才会展示if (selectedIndexList.indexOf(i) > -1) {ele.style = `${baseStyle}opacity:0.7`;} else {ele.style = `${baseStyle}opacity:0`;}});});// 用户点击时,保存用户选中的mask的indexbox.addEventListener("mousedown", (e) => {const { clientX, clientY } = e;const X = box.getBoundingClientRect().left + document.body.scrollLeft;const Y = box.getBoundingClientRect().top + document.body.scrollTop;const x = parseInt(res.width * (clientX - X) / box.getBoundingClientRect().width)const y = parseInt(res.height * (clientY - Y) / box.getBoundingClientRect().height)const index = layers.findIndex((item) => item.matrix?.[y]?.[x]);if (selectedIndexList.indexOf(index) === -1) {//保存点击选中的元素indexselectedIndexList.push(index)}});

box就是上文的layer-box,是各个mask的父组件。layer-box监听鼠标的move事件和click事件,当move到对应的mask上时,将mask展示,移除mask时,隐藏mask。mask在list中是从小到大的顺序,所以遍历匹配mask时,会优先匹配面积小的组件,方便灵活选择。当点击mask的位置时,保存mask在list中的index到selectedIndexList中,方便后续导出保存选择,并高亮展示选中的mask。

选中的mask合成图片
// 存储各个图层图片信息let layers = []// 选择layer的indexconst selectedIndexList = []// 点击保存document.getElementById('save').onclick = function () {const images = [];selectedIndexList.forEach(index => {images.push(layers[index].imageData)})drawing(images)}/*** 图片合成*/function drawing(images) {const canvas = document.getElementById("mergedCanvas");canvas.width = 500;  // 设置canvas宽canvas.height = 500; // 设置canvas高const ctx = canvas.getContext("2d");let loadedImages = 0;images.forEach(function (src) {const img = new Image();img.src = src;img.onload = function () {loadedImages++;// 绘制每张图片到 canvas 上ctx.drawImage(img, 0, 0);// 如果所有图片都加载完成,保存合并后的图片if (loadedImages === images.length) {// 获取图片的像素数据const imageData = ctx.getImageData(0, 0, img.width, img.height);const data = imageData.data;// 转换为黑白效果for (let i = 0; i < data.length; i += 4) {// 将 R、G、B 设置为0data[i] = 0;data[i + 1] = 0;data[i + 2] = 0;}// 将修改后的数据放回 canvasctx.putImageData(imageData, 0, 0);// 导出为 base64 图片const mergedImageBase64 = canvas.toDataURL("image/png");// 如果需要,你可以将mergedImageBase64图片用于其他操作,比如发送到服务器}};});}

当选择完成后,可以点击“保存”按钮,将选择的mask使用canvas生成一个合并后的图片。此处已将合成后的图片转换为黑白蒙版照片,之后可以使用这个合并后的图片进行后续的处理。

根据选中的图层,点击保存后,生成的模板如下图所示。

389768c6055569702819cda63bb1dc49.jpeg

预览效果(https://yuhao1128.github.io/AI-model-mask-select-demo/)、代码详情(https://github.com/yuhao1128/AI-model-mask-select-demo/blob/main/index.html)

使用Stable Diffusion进行后续的处理

由于篇幅的限制,并且这部分网络上以及有很多的介绍资料,就不再本文中进行介绍了,可以参考这篇文章(https://www.uisdc.com/stable-diffusion-24)的介绍尝试体验一下在本地中使用Stable Diffusion的图生图的「重绘蒙版」来进行模特的重新绘制。

也可以在后端部署Stable Diffusion服务中处理模特换装。将前面的模特原图以及生成的蒙版图片,以及其他的SD的图生图功能的参数传给后端的SD服务处理。

除了模特换装的功能,上面的流程还可以应用到物品换背景的功能中。其他的一些智能抠图,智能替换的功能都可以扩展上面的处理流程来实现。

参考链接:

https://github.com/facebookresearch/segment-anything

https://juejin.cn/post/7248903246970503223#heading-2

https://www.uisdc.com/stable-diffusion-24

- END -

关于奇舞团

奇舞团是 360 集团最大的大前端团队,代表集团参与 W3C 和 ECMA 会员(TC39)工作。奇舞团非常重视人才培养,有工程师、讲师、翻译官、业务接口人、团队 Leader 等多种发展方向供员工选择,并辅以提供相应的技术力、专业力、通用力、领导力等培训课程。奇舞团以开放和求贤的心态欢迎各种优秀人才关注和加入奇舞团。

fe6c7c7319f195cbcc2ca1909b9dfe06.jpeg

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/205947.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

基于Java SSM框架+Vue实现汉服文化平台网站项目【项目源码+论文说明】计算机毕业设计

基于java的SSM框架Vue实现汉服文化平台系统演示 摘要 本论文主要论述了如何使用JAVA语言开发一个汉服文化平台网站 &#xff0c;本系统将严格按照软件开发流程进行各个阶段的工作&#xff0c;采用B/S架构&#xff0c;面向对象编程思想进行项目开发。在引言中&#xff0c;作者将…

MTK联发科MT6762/MT6763/MT6765安卓核心板参数规格比较

MT6762安卓核心板 MTK6762安卓核心板是一款工业级高性能、可运行 android9.0 操作系统的 4G智能模块。 CPU&#xff1a;4xCortex-A53 up to 2.0Ghz/4xCortex-A53 up to 1.5GhzGraphics&#xff1a;IMG GE8320 Up to 650MhzProcess&#xff1a;12nmMemory&#xff1a;1xLP3 9…

【虚拟机】Docker基础 【二】

2.2.数据卷 容器是隔离环境&#xff0c;容器内程序的文件、配置、运行时产生的容器都在容器内部&#xff0c;我们要读写容器内的文件非常不方便。大家思考几个问题&#xff1a; 如果要升级MySQL版本&#xff0c;需要销毁旧容器&#xff0c;那么数据岂不是跟着被销毁了&#x…

【攻防世界-misc】CatCatCat

1.下载附件并解压至桌面&#xff0c; 包含一张图片&#xff0c;一个txt文件&#xff0c;将图片复制到kali桌面上&#xff0c;使用strings命令查看该图片内容是否包含flag字符&#xff0c;得到的内容是密码为&#xff1a;catflag 在查看txt文件时&#xff0c;可以看到在文件名命…

Matlab通信仿真系列——随机信号分析

微信公众号上线&#xff0c;搜索公众号小灰灰的FPGA,关注可获取相关源码&#xff0c;定期更新有关FPGA的项目以及开源项目源码&#xff0c;包括但不限于各类检测芯片驱动、低速接口驱动、高速接口驱动、数据信号处理、图像处理以及AXI总线等 本节目录 一、平稳随机过程 1、相…

el-select实现分屏效果

动态绑定class值 &#xff0c;多种判断 :class"type 8 ? home-stye-2 : type 24 ? home-stye-1 : home-stye-3" <div class"home-right-top"><div class"home-right-top-video"><el-row :gutter"20"><el-c…

SpringMvc集成开源流量监控、限流、熔断降级、负载保护组件Sentinel | 京东云技术团队

前言&#xff1a;作者查阅了Sentinel官网、51CTO、CSDN、码农家园、博客园等很多技术文章都没有很准确的springmvc集成Sentinel的示例&#xff0c;因此整理了本文&#xff0c;主要介绍SpringMvc集成Sentinel SpringMvc集成Sentinel 一、Sentinel 介绍 随着微服务的流行&…

docker和docker-compose生产的容器,不在同一个网段,解决方式

在实际项目中&#xff0c;使用docker run xxXx 和docker-compose up -d 不在同一个网段&#xff0c;一个是默认是172.17.x.x, 另一个是172.19.x.x。为解决这个问题需要自定义一个网络&#xff0c;我命名为“my-bridge” 首先熟悉几条命令&#xff1a; docker network ls 或…

UG\NX二次开发 创建对象属性UF_ATTR_set_user_attribute

文章作者:里海 来源网站:里海NX二次开发3000例专栏 简介 创建对象属性UF_ATTR_set_user_attribute,这是一个新函数用于替代UF_ATTR_assign,旧版本NX是用UF_ATTR_assign函数创建、更新属性值,请参照这篇文章《UG\NX二次开发 创建对象属性UF_ATTR_assign》 下面是这个新函数…

什么是requestIdleCallback?和requestAnimationFrame有什么区别?

什么是requestIdleCallback? 我们都知道React 16实现了新的调度策略(Fiber), 新的调度策略提到的异步、可中断&#xff0c;其实就是基于浏览器的 requestIdleCallback和requestAnimationFrame两个API。 在 JavaScript 中&#xff0c;requestIdleCallback 是一个用于执行回调函…

算法通关第十七关黄金挑战——透析跳跃问题

大家好&#xff0c;我是怒码少年小码。 本篇是贪心思想的跳跃问题专题&#xff0c;跳跃问题出现的频率很高。 跳跃游戏 LeetCode 55&#xff1a;给你一个非负整数数组 nums &#xff0c;你最初位于数组的 第一个下标。数组中的每个元素代表你在该位置可以跳跃的最大长度。 …

【数据结构】——排序

&#x1f383;个人专栏&#xff1a; &#x1f42c; 算法设计与分析&#xff1a;算法设计与分析_IT闫的博客-CSDN博客 &#x1f433;Java基础&#xff1a;Java基础_IT闫的博客-CSDN博客 &#x1f40b;c语言&#xff1a;c语言_IT闫的博客-CSDN博客 &#x1f41f;MySQL&#xff1a…

【网络奇遇之旅】:那年我与计算机网络的初相遇

&#x1f3a5; 屿小夏 &#xff1a; 个人主页 &#x1f525;个人专栏 &#xff1a; 计算机网络 &#x1f304; 莫道桑榆晚&#xff0c;为霞尚满天&#xff01; 文章目录 一. 前言二. 计算机网络的定义三. 计算机网络的功能3.1 资源共享3.2 通信功能3.3 其他功能 四. 计算机网络…

Mysql 递归查询子类Id的所有父类Id

文章目录 问题描述先看结果表结构展示实现递归查询集合查询结果修复数据 问题描述 最近开发过程中遇到一个问题,每次添加代理关系都要去递归查询一下它在不在这个代理关系树上.很麻烦也很浪费资源.想着把代理关系的父类全部存起来 先看结果 表结构展示 表名(t_agent_user_rela…

如何把ipa文件(iOS安装包)安装到iPhone手机上? 附方法汇总

文章目录 概要整体架构流程技术名词解释技术细节小结 概要 技术细节 目录 Appuploader 常见错误及解决方法 苹果APP安装包ipa如何安装在手机上&#xff1f;很多人不知道怎么把ipa文件安装到手机上&#xff0c;这里就整理了苹果APP安装到iOS设备上的方式&#xff0c;仅供参考 苹…

Linux基础指令

1.ls指令 【语法】ls 目录/普通文件 对于目录&#xff0c;列出目录中的所有文件对于普通文件&#xff0c;列出文件的基本属性 选项&#xff1a; -l 详细列出文件的属性-a 列出当前目录下的文件和隐藏文件-i 显示文件的索引信息-R 以递归的方式显示目录下的文件 1.1 [ls -l…

sql注入靶场

第一关&#xff1a; 输入&#xff1a;http://127.0.0.1/sqli-labs-master/Less-1/?id1 http://127.0.0.1/sqli-labs-master/Less-1/?id1%27 http://127.0.0.1/sqli-labs-master/Less-1/?id1%27-- 使用--来闭合单引号&#xff0c;证明此处存在字符型的SQL注入。 使用order …

《程序员考公指南》:零基础到上岸的完整攻略 | 开源日报 No.82

mastodon/mastodon Stars: 44.2k License: AGPL-3.0 Mastodon 是一个免费、开源的社交网络服务器&#xff0c;基于 ActivityPub。用户可以在 Mastodon 上关注朋友并发现新朋友&#xff0c;并且可以发布链接、图片、文字和视频等内容。所有的 Mastodon 服务器都能互操作成为联邦…

第五届全国高校计算机能力挑战赛-程序设计挑战赛(C语言模拟题)

1、已有定义“int a[10]{1,2},i0;”&#xff0c;下面语句中与“ a[i]a[i1],i;”等价的是()。 A. a[i]a[i1]; B. a[i]a[i]; C. a[i]a[i1]; D. i,a[i-1]a[i]; 2、两次运行下面的程序&#xff0c;如果从键盘上分别输入6和4&#xff0c;则输出结果是(&#xff09;。 A. 7和5 …